- Tableview performance optimization method overview
- TableViewCell reuse
- Cache cell height
- The rounded optimization
- Asynchronous rendering
- Other optimization
- Comparison of some optimization schemes
Tableview performance optimization method overview
- Lazy loading of TableView and cell reuse
- High cache (because
heightForRowAtIndexPath:
Is the most frequently called method)- When cell rowHeight is fixed, use fixed rowHeight self.tableview.rowheight = 88;
- If the row height of a cell is not fixed, it is calculated based on the content and cached. The first time it will definitely be computed, and the subsequent use of caching will avoid multiple computations; The height calculation method is usually written in a custom cell. When called, it can be used either in the proxy method that sets the cell height or in the custom model (and when used, it is handled with the GET method).
- The data processing
(1) Use the correct data structure to store data; (2) Data should be refreshed by partial sections or cellRow as far as possible to avoid reloadData; (3) When a large amount of data is operated, different steps are used for processing, avoiding direct operation in the main thread; (4) Cache the request results; 4. Asynchronously loading images: the use of SDWebImage (1) Use different steps thread processing, and then return to the main thread operation; (2) Image caching to avoid multiple processing operations; (3) When filleting images, set the layer shouldRasterize attribute to YES to transfer the load to the CPU; (1) In the sliding operation, only the cell contents within the target range will be displayed, and those displayed beyond the target range will be cleared; (2) During the sliding process, the display picture is not loaded until it stops; 6. View level (1) Reduce the number of subviews, and the customized subviews can be integrated into a whole subview as soon as they form a whole; (2) Use drawRect to draw (that is, transfer part of the RENDERING of GPU to CPU) or CALayer to draw text or pictures. When implementing drawRect method, we should pay attention to reducing redundant drawing operations. Its parameter RECT is the area we need to draw, and we do not need to draw the area outside the scope of RECT, otherwise it will consume considerable resources. (3) asynchronous rendering, and set properties. Self layer. DrawsAsynchronously = YES; (Encounter complex interface, performance bottleneck, may be a breakthrough); (4) Define a type of Cell (as few as possible) and make good use of hidden subviews. (5) Try to make the opaque attribute of all views YES, including Cell itself, to improve the rendering speed of view (avoid useless alpha channel composition, Reduce GPU load) (6) Avoid gradient, image scaling (7) use shadowPath to draw shadows (8) avoid cellForRowAtIndexPath: If you need to use it, just use it once and cache the result (9) cellForRowAtIndexPath Do not do time-consuming operations such as not reading/writing files; (10) When we add system controls to the Cell, in fact, the system will call the underlying interface for drawing. When we add a large number of controls, It consumes a lot of resources and also affects rendering performance. Using the default UITableViewCell and adding controls to its ContentView can be very performance consuming. So the best thing for now is to inherit the UITableViewCell and rewrite the drawRect method. (11) When we need rounded corners, Use ShadowPath to specify the layer shadow. Use asynchronous Layer rendering (AsyncDisplayKit) to set the opaque value of the layer to YES Reduce complex layer composition try to use image resources that do not contain alpha channels try to set the size of layer to an integer value and directly ask the artist to cut the image into rounded corners for display. This is the most efficient solution. In many cases, users upload images for display. You can have the server handle rounded corners using code to manually generate a rounded Image set to the View to display, using UIBezierPath (CoreGraphics framework) to draw rounded corners
TableViewCell reuse
- Introduction to tableViewCell reuse
There’s a pool of cells inside the tableView, and that’s the pool of cells that you created earlier. Memory rich will save some UITableViewCell objects into the cell pool and quickly return them when needed, without creating them. When memory is tight, the cell pool automatically cleans up some superfluous UITableViewCell objects. This internal controls how many cells there are. Note: It is possible to bind data or add subviews to a retrieved cell, so if necessary, clear the data (such as the label border) so that it displays the correct content. 2. TableviewCell multiplexing method dequeueReusableCellWithIdentifier: forIndexPath: (iOS6 introduction)
// It must be used with the register method, otherwise the cell returned may be nil, Will crash [slef myTableView registerClass: [MyCell class] forCellReuseIdentifier: NSStringFromClass ([MyCell class])]; MyCell* cell = [tableView dequeueReusableCellWithIdentifier:NSStringFromClass([MyCell class]) forIndexPath:indexPath];Copy the code
- Register different types of cells or do not reuse them. The following uses non-reuse as an example
@property (nonatomic, strong) NSMutableDictionary *cellDic; NSString *identifier = [_cellDic objectForKey:[NSString stringWithFormat:@"%@", indexPath]]; // If the retrieved unique identifier does not exist, initialize the unique identifier and store it in the dictionary, Cell if (Identifier == nil) {identifier = [NSString stringWithFormat:@"%@%@", @" Cell ", [NSString stringWithFormat:@"%@", indexPath]]; [_cellDic setValue:identifier forKey:[NSString stringWithFormat:@"%@", indexPath]]; / / registered Cell [self tableview registerClass: [MyCell class] forCellWithReuseIdentifier: identifier]; } MyCell *cell = [tableView dequeueReusableCellWithReuseIdentifier:identifier forIndexPath:indexPath];Copy the code
- If you look at the UITableView header file, you will find NSMutableArray *visiableCells, and NSMutableDictionary *reusableTableCells. VisiableCells are used to store cells that are currently displayed in the UITableView, and reusableTableCells are used to store cells that have been cached with ‘identify’. When UITableView scrolls, reusableTableCells will check if there is a cached cell by identify. If there is a cached cell by identify, it will not initialize it. (at the beginning of the TableView displays, reusableTableCells is empty, then the TableView dequeueReusableCellWithIdentifier: CellIdentifier returns nil. The start of the cell is through the [[UITableViewCell alloc] initWithStyle: UITableViewCellStyleDefault reuseIdentifier: CellIdentifier] to create, And cellForRowAtIndexPath is just the number of calls to the maximum display cell.)
Cache cell height
- If frame is used, add a cellH property to the model and assign the height to cellH when fetching data.
- If using AutoLayout, create cells with the same layout, calculate the height and cache it.
@property (nonatomic, strong) NSMutableDictionary *heightAtIndexPath; #pragma mark - UITableViewDelegate -(CGFloat)tableView:(UITableView *)tableView estimatedHeightForRowAtIndexPath:(NSIndexPath *)indexPath { NSNumber *height = [self.heightAtIndexPath objectForKey:indexPath]; if(height){ return height.floatValue; }else { return 100; } } - (void)tableView:(UITableView *)tableView willDisplayCell:(UITableViewCell *)cell forRowAtIndexPath:(NSIndexPath *)indexPath { NSNumber *height = @(cell.frame.size.height); [self.heightAtIndexPath setObject:height forKey:indexPath]; }Copy the code
FD implementation:fd_heightForCellWithIdentifier: configuration:
The Identifier and Configuration block method provides a Template Layout cell that is identical to the cell layout and passes it infd_systemFittingHeightForConfiguratedCell:
This private method returns the calculated height. The main technology used is Runtime.
Off-screen rendering
In OpenGL, there are two ways to render a GPU Screen: on-screen Rendering: The GPU renders On the Screen buffer currently used for display. Off-screen Rendering: When the GPU creates a new buffer outside the current Screen buffer to render.
Off-screen rendering is expensive compared to the current screen rendering, mainly in two ways:
- Create a new buffer. To render off-screen, first create a new buffer.
- Context switching: The whole process of off-screen rendering requires multiple context switching: first, it switches from the current Screen to the off-screen; After the off-screen rendering is complete, the rendering results of the off-screen buffer are displayed on the screen. It is necessary to switch the context from off-screen to the current screen. Context switching is costly.
Off-screen render trigger condition
- custom drawRect: (any, even if you simply fill the background with color)
- CALayer mask
- CALayer shadow
- any custom drawing using CGContext
The specific performance is mask, shadow, shouldRasterize, edge antialiasing, group opacity, rounded corners for complex shapes, and gradient
- The CPU and GPU:
The CPU, which does view-related calculations and tells the GPU how to draw; GPU, graphics drawing, rendering and other work;
The rounded optimization
- Optimization 1: Draw a rounded corner using the Bezier curve UIBezierPath and Core Graphics framework
UIImageView *imageView = [[UIImageView alloc]initWithFrame:CGRectMake(100, 100, 100, 100)]; imageView.image = [UIImage imageNamed:@"myImg"]; / / to draw the imageView UIGraphicsBeginImageContextWithOptions (imageView. Bounds. The size, NO, 1.0); / / using bezier draw a circular diagram [[UIBezierPath bezierPathWithRoundedRect: imageView. Bounds cornerRadius: imageView. Frame. The size, width] addClip]; [imageView drawRect:imageView.bounds]; imageView.image = UIGraphicsGetImageFromCurrentImageContext(); / / end drawing UIGraphicsEndImageContext (); [self.view addSubview:imageView];Copy the code
- Optimization 2: Use CAShapeLayer and UIBezierPath to set rounded corners
UIImageView * imageView = [[UIImageViewalloc] initWithFrame: CGRectMake (100100100100)]. imageView.image = [UIImageimageNamed:@"myImg"]; UIBezierPath *maskPath = [UIBezierPathbezierPathWithRoundedRect:imageView.boundsbyRoundingCorners:UIRectCornerAllCornerscornerRadii:imageView.bou nds.size]; CAShapeLayer *maskLayer = [[CAShapeLayer alloc]init]; Masklayer. frame=imageView.bounds; // MaskLayer. path= maskpath. CGPath; imageView.layer.mask=maskLayer; [self.viewaddSubview:imageView];Copy the code
For scheme 2, it needs to be explained that CAShapeLayer inherits from CALayer and can use all the attribute values of CALayer. CAShapeLayer needs bezier curves to be used together to make sense. Using CAShapeLayer(belonging to CoreAnimation) and bezier curves can achieve drawRect that is not in view (which is derived from CoreGraphics and goes through CPU consumption) CAShapeLayer animation rendering is directly submitted to the GPU of the mobile phone. Compared with the VIEW drawRect method using CPU rendering, it is highly efficient and can greatly optimize the memory usage. In general, CAShapeLayer has less memory consumption and faster rendering speed. Therefore, optimization scheme 2 is recommended.
Asynchronous rendering
System drawing flow chart
- Create a backing Store inside CALayer (CGContextRef)();
- Determine whether layer has proxy; There are proxies: call delegete’s drawLayer:inContext, and then do some drawing in [UIView drawRect] in the appropriate actual callback proxy; No proxy: Call Layer’s drawInContext method,
- Layer uploads backingStore to GPU to end the system drawing process.
UIView drawing flowchart
- UIView calls setNeedsDisplay, but it doesn’t draw the view immediately;
- After UIView calls setNeedDisplay, the system calls the setNeedsDisplay method of the layer corresponding to view.
- Call CALayer’s display method when the current runloop is about to end;
- The runloop is about to end, starting the view drawing process;
Asynchronous rendering
- Asynchronously drawn entry at [layer.delegate displayLayer]
- In the process of asynchronous drawing, the proxy is responsible for generating the corresponding bitmap.
- Assign the bitmap to the layer.content property;
- At some point setNeedsDisplay is called;
- Call [CALayer display] when runloop is about to end
- If the proxy implements dispalyLayer it will call this method to do the asynchronous drawing in the child thread;
- Work done in child threads: create context, draw controls, generate images;
- Go to the main thread and set layer.contents to display the generated view on top of the layer.
#import <UIKit/UIKit.h> NS_ASSUME_NONNULL_BEGIN @interface AsyncLabel : UIView @property (nonatomic, copy) NSString *asynText; @property (nonatomic, strong) UIFont *asynFont; @property (nonatomic, strong) UIColor *asynBGColor; @end NS_ASSUME_NONNULL_END #import "AsyncLabel.h" #import <CoreText/CoreText.h> @implementation AsyncLabel - (void)displayLayer (CALayer *)layer {/** Except in drawRect, Other places need create context for [https://www.jianshu.com/p/86f025f06d62] CoreText usage summary: [https://www.cnblogs.com/purple-sweet-pottoes/p/5109413.html] * / CGSize size = self. The bounds. Size;; CGFloat scale = [UIScreen mainScreen].scale; / / / asynchronous mapping: switch to the child thread dispatch_async (dispatch_get_global_queue (0, 0), ^ {UIGraphicsBeginImageContextWithOptions (size, NO, scale); / / / get the current context CGContextRef context = UIGraphicsGetCurrentContext (); / / / will coordinate inversion CGContextSetTextMatrix (context, CGAffineTransformIdentity); /// Move the text along the Y axis CGContextTranslateCTM(context, 0, sie.height); // Invert the text to the context coordinate system CGContextScaleCTM(context, 1.0, -1.0); CGMutablePathRef path = CGPathCreateMutable(); CGPathAddRect(path, NULL, CGRectMake(0, 0, size.width, size.height)); / / / create a need to draw text NSMutableAttributedString * attStr = [[NSMutableAttributedString alloc] initWithString: self. AsynText]; [attStr addAttribute:NSFontAttributeName value:self.asynFont range:NSMakeRange(0, self.asynText.length)]; [attStr addAttribute:NSBackgroundColorAttributeName value:self.asynBGColor range:NSMakeRange(0, self.asynText.length)]; CTFramesetterRef CTFramesetterRef frameSetter = CTFramesetterCreateWithAttributedString((CFAttributedStringRef)attStr); CTFrameRef frame = CTFramesetterCreateFrame(frameSetter, CFRangeMake(0, attStr.length), path, NULL); CTFrameDraw(frame, context); UIImage *getImg = UIGraphicsGetImageFromCurrentImageContext(); UIGraphicsEndImageContext(); /// dispatch_async(dispatch_get_main_queue(), ^{self.layer.contents = (__bridge id) getimg.cgimage; }); }); } @end #import "ViewController.h" #import "AsyncLabel.h" @interface ViewController () @end @implementation ViewController - (void)viewDidLoad { [super viewDidLoad]; AsyncLabel *asLabel = [[AsyncLabel alloc] initWithFrame:CGRectMake(50, 100, 200, 200)]; asLabel.backgroundColor = [UIColor cyanColor]; asLabel.asynBGColor = [UIColor greenColor]; asLabel.asynFont = [UIFont systemFontOfSize:16 weight:20]; AsLabel. AsynText = @" asynText "; [self.view addSubview:asLabel]; // The displayLayer method [aslabel. layer setNeedsDisplay] will not be triggered if it is not called; } @endCopy the code
Other optimization
- Child threads process data asynchronously
- (void)loadData{dispatch_async(dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0), ^{// Handle data coding... // dispatch_async(dispatch_get_main_queue(), ^{[self.mainTableView reloadData]; }); });Copy the code
- Don’t do extra drawing work
When you implement drawRect:, its rect parameter is the region to draw, and anything outside that region is not to be drawn. For example, in the above example, CGRectIntersectsRect, CGRectIntersection or CGRectContainsRect can be used to determine whether image and text need to be drawn, and then call the drawing method.
- As shown in the figure, the content displayed by this label is concatenated by two parameters of model (time and kilometers). We are used to assigning values like this in the set method of model in cell
NSDateFormatter* formatter = [[NSDateFormatter alloc] init]; formatter.dateStyle = NSDateFormatterMediumStyle; formatter.timeStyle = NSDateFormatterShortStyle; [formatter setDateFormat:@" YYYY "]; NSDate* date = [NSDate dateWithTimeIntervalSince1970:[model.licenseTime intValue]]; NSString* licenseTimeString = [formatter stringFromDate:date]; NSString *travelMileageString = (model. TravelMileage! = nil && ! [model.travelMileage isEqualToString:@""]) ? [NSString stringWithFormat:@"%@ million km ",model. TravelMileage] :@" no "; / / assignment to label text self. CarDescribeLabel. Text = [nsstrings stringWithFormat: @ "% @ / %@",licenseTimeString,travelMileageString];Copy the code
- As the TableView scrolls, these objects are created back and forth, and the calculation is done in the main thread.
We can move these operations to step 2 (dictionary to model), calculate the content that the label needs to display, save it as an attribute in the model, and use it directly when needed. Here’s an example of the caching idea: – (CGFloat)tableView (UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath {return 15.0 + 80.0 + 15.0; } static float ROW_HEIGHT = 15.0 + 80.0 + 15.0; – (CGFloat)tableView:(UITableView *)tableView heightForRowAtIndexPath:(NSIndexPath *)indexPath { return ROW_HEIGHT; } Of course, this is not to reduce the creation of objects, but to reduce the number of computations, less frequent calls to the logic in the method, and thus faster.
Some optimization scheme comparison
- By Autolayout Autolayout + AutomaticDimension: layout, the height of the Cell adaptive UITableViewAutomaticDimension use system. This implementation is easy to use, but when the TableView slides quickly, it will drop frames, special blocks.
- Autolayout + CountHeight: Autolayout layout, the height of the Cell is calculated by the subline itself, better than the first implementation, but the frame drop is also more serious.
- FrameLayout + CountHeight: Frame layout, Cell height calculation step in child thread. A smooth compromise.
- YYKit + CountHeight: use YYKit control, and use Frame layout and Cell height calculation. This approach is better than the above solution because some of the controls in YYKit are optimized.
- AsyncDisplayKit + CountHeight: Uses the relevant Note provided in AsyncDisplayKit to replace the system’s native controls. This implementation is the most smooth among the five implementations.
VVeboTableViewDemo performance optimization -UITableView optimization using iOS to keep the interface smooth tips