ios性能优化-卡顿优化和耗电优化

1 卡顿产生的原因及优化

产生卡顿是由于屏幕的成像显示导致,而屏幕画面的显示离不开手机的CPU和GPU;

CPU:(Central Processing Unit 中央处理器)
对象的创建和销毁,对象属性的调整,布局的计算,文本的布局计算和排版,图片格式的转换和解码,图像的绘制(Core Graphics)

GPU: (Graphics Processing Unit 图形处理器)
纹理的绘制

iOS是双帧缓存机制,有前帧缓存,后帧缓存

image.png

1.1屏幕成像显示的过程是:

  • CPU先计算出图像的布局,大小,位置等信息;(CPU计算出来的数据是不能直接显示到屏幕上的)
  • GPU将CPU计算的数据,渲染到帧缓存中;
  • 要显示图像的时候,视频控制器从帧缓存中读取图像,显示到屏幕上;

1.2 屏幕成像显示的原理:

iPhone的刷帧频率是 60 FPS,也就是每秒显示60帧数据;

每帧图像显示的时间间隔是: 1000ms / 60 fps = 16ms;

如图:


屏幕成像

屏幕在显示一帧数据的时候:

  • 会先发送一条垂直同步信号
  • 然后会 从上至下 发送水平同步信号,填充整个屏幕,显示这一帧的数据;

重点:每隔16ms就会显示下一帧数据,接收到 垂直同步信号 代表开始显示下一帧的内容

1.3 显示和卡顿产生的根本原因:

之前介绍,屏幕成像在CPU计算和GPU渲染到帧缓存区之后,再由视频控制器读取并显示到屏幕上。

如下图所示:

1、2、3、4、5 代表5帧数据的显示流程,
其中红色箭头代表CPU计算所用时间,蓝色箭头代表GPU渲染所用时间;

image.png
  • 第一帧 1:CPU和GPU所花的时间 ==16ms,所以在 垂直同步信号到来的时候,帧缓存中有完整的数据,正常展示;
  • 第二帧 2:CPU和GPU所花的时间 < 16ms, 超前将要显示的内容绘制到帧缓存中,正常展示;
  • 第三帧 3:CPU和GPU所花的时间 > 16ms, 16ms内这一帧的数据还没渲染完成,垂直同步信号已经到来,帧缓存中的数据不全,这一帧会继续显示上一帧(第二帧)的内容
    所以这就是卡顿的原因;
  • 第四帧 4:在这一次的显示中,第三帧的内容CPU和GPU渲染刚完成,当垂直信号到来时,去帧缓存中去读取并直接显示 第三帧 的内容;
  • 第五帧 5:同第二帧,正常展示;

由上图直接展示了卡顿产生的 根本原因

在一帧显示的频率16ms中,如果CPU和GPU没有将要显示的内容渲染到帧缓存中,当前垂直同步信号到来的时候,就会显示上一帧的内容;

这一帧的内容,会在下一个周期16ms后,垂直同步信号再次到来的时候,显示到屏幕上。

1.4 解决卡顿的方式CPU和GPU:

CPU:

1、使用轻量级的对象:比如不用点击的地方,使用CALayer代替UIView;

2、不要频繁的修改属性:frame,bounds,transfrom等,这些都需要CPU的计算;

3、尽量提前计算好布局:计算好frame,bounds等,一次性修改,不要多次修改;

4、使用AutoLayout比直接设置frame消耗更多的资源;

5、图片的size最好和UIImageView的size保持一致,这样就不用耗费CPU资源去进行缩放操作;

6、控制线程的最大并发数量:比如说3,不要无限制的开辟新的线程;

7、尽量耗时操作放到子线程:

  • 文本的计算(高度),绘制(排版)等
    // 文字计算
    [@"text" boundingRectWithSize:CGSizeMake(100, MAXFLOAT) options:NSStringDrawingUsesLineFragmentOrigin attributes:nil context:nil];
        
    // 文字绘制
    [@"text" drawWithRect:CGRectMake(0, 0, 100, 100) options:NSStringDrawingUsesLineFragmentOrigin attributes:nil context:nil];
  • 图片的解码、绘制

正常图片的展示:imageView.image = [UIImage imageNamed:@"test.png"];

    UIImageView *imageView = [[UIImageView alloc] init];
    imageView.frame = CGRectMake(100, 100, 100, 56);
    imageView.image = [UIImage imageNamed:@"test.png"];
    [self.view addSubview:imageView];  

其实正常图片的显示,不是直接展示到屏幕上的,需要解码成能够展示的二进制数据,而这个解码的过程,可以异步的放到子线程中去做:

- (void)viewDidLoad {
    [super viewDidLoad];
    
    UIImageView *imageView = [[UIImageView alloc] init];
    imageView.frame = CGRectMake(100, 100, 100, 56);
    [self.view addSubview:imageView];
    self.imageView = imageView;
    
    [self image]; //异步解码图片,解码成功后再回主线程展示
}

- (void)image{
    dispatch_async(dispatch_get_global_queue(0, 0), ^{
        // 获取CGImage
        CGImageRef cgImage = [UIImage imageNamed:@"test.png"].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;
        });
    });
}

其实就是将image转化成CGImage,然后将CGImage解码,首先创建一个上下文,通过drawImage方法将image画到上下文context完成解码操作,然后从上下文获取解码后的图片;

GPU:

1、尽量减少视图的数量和层级:多层次的视图绘制更占用GPU资源;

2、尽量避免短时间大量图片的显示:可以合成为一张图片展示;

3、GPU能处理的图片的最大尺寸是4096x4096,尽量不要超过这个尺寸;

4、减少透明视图的使用 alpha < 1,
重叠部分:有透明度:需要混合计算;不透明:计算一次(最上层的颜色)

5、避免离屏渲染:

离屏渲染

  • 当前屏幕渲染:(On-Screen Rendering)在当前显示的屏幕缓冲区进行操作;

  • 离屏渲染: (Off-Screen Rendering)在当前屏幕缓冲区以外,开辟一个新的缓冲区;

离屏渲染消耗性能的原因:

  • 需要开辟新的缓冲区;

  • 需要多次切换上下文状态:从当前屏幕(On-Screen)切换到离屏(Off-Screen),等离屏渲染结束以后,又要从离屏切换到当前屏幕;

哪些操作会触发离屏渲染?

  • 1、光栅化:layer.shouldRasterize = YES;

  • 2、遮罩:layer.mask;

  • 3、圆角,同时设置layer.masksToBounds = YES、layer.cornerRadius大于0;

    解决办法:考虑通过CoreGraphics绘制裁剪圆角,或者叫美工提供圆角图片;

  • 4、阴影,layer.shadowXXX;
    如果设置了layer.shadowPath就不会产生离屏渲染(不设置路径默认是围绕这个view)

卡顿检测:

平时所说的“卡顿”主要是因为在主线程执行了比较耗时的操作

可以添加Observer到主线程RunLoop中,通过监听RunLoop状态切换的耗时,以达到监控卡顿的目的

耗电优化

耗电的主要来源:

  • CPU处理计算
  • 网络请求
  • 定位
  • 图形的处理

耗电优化的处理:

  • 1、 尽可能减少CPU和GPU的消耗;

  • 2、 优化I/O操作:

    • 尽量不要频繁的读写小数据,可以批量一次性写入
    • 读写大量重要数据时,考虑用dispatch_io,其提供了基于GCD的异步操作文件I/O的API。用dispatch_io系统会优化磁盘访问
    • 数据量比较大的,建议使用数据库(比如SQLite、CoreData)
  • 3、网络优化:

    • 减少、压缩网络数据
    • 如果多次请求的结果是相同的,尽量使用缓存
    • 使用断点续传,否则网络不稳定时可能多次传输相同的内容
    • 网络不可用时,不要尝试执行网络请求
    • 让用户可以取消长时间运行或者速度很慢的网络操作,设置合适的超时时间
    • 批量传输,比如,下载视频流时,不要传输很小的数据包,直接下载整个文件或者一大块一大块地下载。如果下载广告,一次性多下载一些,然后再慢慢展示。如果下载电子邮件,一次下载多封,不要一封一封地下载
  • 4、定位优化

    • 如果只是需要快速确定用户位置,最好用CLLocationManager的requestLocation方法。
    • 定位完成后,会自动让定位硬件断电
    • 如果不是导航应用,尽量不要实时更新位置,定位完毕就关掉定位服务
    • 尽量降低定位精度,比如尽量不要使用精度最高的kCLLocationAccuracyBest
    • 需要后台定位时,尽量设置pausesLocationUpdatesAutomatically为YES,如果用户不太可能移动的时候系统会自动暂停位置更新
    • 尽量不要使用startMonitoringSignificantLocationChanges,优先考虑startMonitoringForRegion:
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351