avatar

Catalog
iOS性能优化03-卡顿优化01-CPU
  • 尽量使用轻量级的对象,比如用不到事件处理的地方,可以考虑使用CALayer取代UIView

  • 不要频繁的调用UIView的相关属性,比如frame、bounds、transform等属性,尽量减少不必要的修改

  • 尽量提前计算好布局,在有需要时一次性调整对应的属性,不要多次修改属性

  • Autolayout会比直接设置frame消耗更多的CPU资源

  • 图片的size,最好刚好跟UIImageView的size保持一致

  • 控制一下线程的最大并发数量

  • 尽量把耗时的操作放到子线程里(比如文本处理图片处理)

    • 文本处理 (尺寸计算、绘制)

      Code
      1
      2
      // 文字计算
      [@"text" boundingRectWithSize:CGSizeMake(100, MAXFLOAT) options:NSStringDrawingUsesLineFragmentOrigin attributes:nil context:nil];
      Code
      1
      2
      // 文字绘制
      [@"text" drawWithRect:CGRectMake(0, 0, 100, 100) options:NSStringDrawingUsesLineFragmentOrigin attributes:nil context:nil];
- 图片处理(解码、绘制)

我们一般都会用```imageView.image = [UIImage imageNamed:@"***"];```这种方式来获取图片,但这种方式是系统负责解码,而系统解码默认又是在主线程所以如果你的图片数据很大的时候就可能阻塞主线程从而造成卡顿,所以我们一般都是提前解码并且在子线程里进行,这样主线程就少了解码的操作到时候直接把解码的图片数据拿来用就好了,其实我们现在用的一些第三方图片处理库都有异步解码的功能了,我们来看一个例子(也是从三方库里拷贝过来的)代码有点多,其实核心步骤就是:

 1.获取到图片的CGImage

2.解码(解码的步骤就是)

- 2.1 创建位图上下文

- 2.2 把cgImage draw到上下文里,这样就完成了解码操作

3.从上下文里获取刚解码过的图片

4.最好再把解码过的图片包装成UIImage

5.回到主线程的时候我们直接用解码过的图片就OK了(这样解码的操作都是在子线程里做的并不会阻塞主线程)

下面我们来看看具体的代码实现

- (void)image
{
    UIImageView *imageView = [[UIImageView alloc] init];
    imageView.frame = CGRectMake(100, 100, 100, 56);
    [self.view addSubview:imageView];
    self.imageView = imageView;

    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        // 获取CGImage
        CGImageRef cgImage = [UIImage imageNamed:@"***"].CGImage;

        // alphaInfo
        CGImageAlphaInfo alphaInfo = CGImageGetAlphaInfo(cgImage) & kCGBitmapAlphaInfoMask;
        BOOL hasAlpha = NO;
        if (alphaInfo == kCGImageAlphaPremultipliedLast ||
            alphaInfo == kCGImageAlphaPremultipliedFirst ||
            alphaInfo == kCGImageAlphaLast ||
            alphaInfo == kCGImageAlphaFirst) {
            hasAlpha = YES;
        }

        // bitmapInfo
        CGBitmapInfo bitmapInfo = kCGBitmapByteOrder32Host;
        bitmapInfo |= hasAlpha ? kCGImageAlphaPremultipliedFirst : kCGImageAlphaNoneSkipFirst;

        // size
        size_t width = CGImageGetWidth(cgImage);
        size_t height = CGImageGetHeight(cgImage);

        // context
        CGContextRef context = CGBitmapContextCreate(NULL, width, height, 8, 0, CGColorSpaceCreateDeviceRGB(), bitmapInfo);

        // draw
        CGContextDrawImage(context, CGRectMake(0, 0, width, height), cgImage);

        // get CGImage
        cgImage = CGBitmapContextCreateImage(context);

        // into UIImage
        UIImage *newImage = [UIImage imageWithCGImage:cgImage];

        // release
        CGContextRelease(context);
        CGImageRelease(cgImage);

        // back to the main thread
        dispatch_async(dispatch_get_main_queue(), ^{
            self.imageView.image = newImage;
        });
    });
}

喜欢的朋友可以扫描关注我的公众号(多谢支持)

Author: 木子召
Link: https://lizhaobomb.github.io/2020/02/28/iOS%E6%80%A7%E8%83%BD%E4%BC%98%E5%8C%9603-%E5%8D%A1%E9%A1%BF%E4%BC%98%E5%8C%9601-CPU/
Copyright Notice: All articles in this blog are licensed under CC BY-NC-SA 4.0 unless stating additionally.
Donate
  • 微信
    微信
  • 支付寶
    支付寶

Comment