2017/9/2 20:36:12当前位置推荐好文新浪微博浏览文章

iOS中有的时候我们控件要做成圆形 或者是切成圆角,这个时候我们一般都会使用.layer.cornerRadius -> clipsToBounds = YES 的属性来切,这样完全能达到我们的效果,但是如果一个界面上需要切圆角的控件很多,并且列表很长的时候,尤其是像tableView那样如果每一个cell上都有大量的控件需要切,那么就会非常卡顿,帧数严重下降 。其实原因就是这样设置会触发离屏渲染,比较消耗性能。注意:png图片UIImageView处理圆角是不会产生离屏渲染的。(ios9.0之后不会离屏渲染,ios9.0之前还是会离屏渲染)。这里先说下离屏渲染:

iOS的渲染机製:

CPU 计算好显示内容提交到 GPU,GPU 渲染完成后将渲染结果放入帧缓冲区,随后视频控製器会逐行读取帧缓冲区的数据,经过可能的数模转换传递给显示器显示。GPU荧幕渲染有以下两种方式:
On-Screen Rendering
意为抵前荧幕渲染,指的是GPU的渲染操作是在抵前用于显示的荧幕缓冲区中进行。
Off-Screen Rendering
意为离屏渲染,指的是GPU在抵前荧幕缓冲区以外新开拓一个缓冲区进行渲染操作。

由以上可以看出离屏渲染需要重新开拓新的缓存空间,必定要愈加消耗资源。
通过查资料目前知道了设置了以下属性时,都会触发离屏绘製:
shouldRasterize(光栅化)
masks(遮罩)
shadows(阴影)
edge antialiasing(抗锯齿)
group opacity(不透明)
复杂形状设置圆角等
渐变

我用一个现有的小DEMO来测试下,因为这个demo中未至于切圆角,但是有阴影,一样可以出发离屏渲染,所以效果是一样的,在tableView中的自定义cell类中我设置了阴影如图:
使用Instruments测试得到抵前帧数在二三十左右

有齐荧幕是亦出现了黄色图层
所有黄色的高亮的图层都进行了离屏渲染,亦表示这些图层存在着问题,影响性能。而且帧数过低亦会直接影响我们对APP的体验。
怎样避免离屏渲染:
  • 方法一:使用光栅化,.layer.shouldRasterize = YES; -> .layer.rasterizationScale=[UIScreen mainScreen].scale;设置光栅化,可以使离屏渲染的结果缓存到内存中存为位图,使用的时候直接使用缓存,节省了一直离屏渲染损耗的性能。但是如果layer及sublayers常常改变的话,它就会一直不停的渲染及删除缓存重新创建缓存,所以这种情况下建议不要使用光栅化,这样亦是比较损耗性能的。

  • 方法二:最简单的方法,就是图片不作处理,直接覆盖一张中间为圆形透明的图片(推荐使用)这种方法GPU计算多层的混合渲染blending亦是会消耗一点性能的,但比第一种方法还是好上很多的

  • 方法三:Core Graphics绘製圆角。这种方式GPU损耗最低,可以用UIimageView添加个点击手势抵做UIButton使用。
    UIGraphicsBeginImageContextWithOptions(avatarImageView.bounds.size, NO, [UIScreen mainScreen].scale);
    [[UIBezierPath bezierPathWithRoundedRect:avatarImageView.bounds cornerRadius:50] addClip];[image drawInRect:avatarImageView.bounds];
    avatarImageView.image = UIGraphicsGetImageFromCurrentImageContext();UIGraphicsEndImageContext();
    这段方法可以写在SDWebImage的completed回调里,在主线程异步绘製。亦可以封装到UIImageView里,写了个DSRoundImageView。后台线程异步绘製,不会阻塞主线程。目前这种方法只用在imageView上,其他地方呢不知道怎么用,不过这种方法会耗费大量的CPU资源和佔用内存,本人不太建议使用。

最后给大家测试下,因为我之前的demo未至于切圆角,而是使用的阴影,所以我就用第一种方法给大家测一下。

然后运行,打开Instruments,然后看荧幕
图层都是绿色没问题,我们再来看帧数,几乎到60 ,效果很明显。

由此可见以后大家再切圆角的时候多注意下,刚才就研究了这么多,有什么问题及时与我交流吧。。。
网友评论