iOS的离屏渲染及OpenGL渲染架构图分析

  • 离屏渲染

    • 什么是离屏渲染
      如果要在显示屏上显示内容,我们至少需要一块与屏幕像素数据量一样大的frame buffer,作为像素数据存储区域,而这也是GPU存储渲染结果的地方。如果有时因为面临一些限制,无法把渲染结果直接写入frame buffer,而是先暂存在另外的内存区域,之后再写入frame buffer,那么这个过程被称之为离屏渲染

      离屏渲染(Offscreen RenderingGPU在当前屏幕缓冲区以外开辟了一个缓冲区进行操作。

      渲染结果先经过了offscreen buffer,再到frame buffer.png

      app进行额外的渲染和合并->offscreen Buffer组合->frame Buffer->屏幕,离屏渲染一个额外的存储空间(offscreen Buffer)渲染完成写入frame Buffer,离屏渲染的大小限制为屏幕像素点的2.5倍。

    • 离屏渲染的性能
      不开启离屏渲染的逻辑为帧缓冲区(双缓存区)直接显示到屏幕,画完即可丢弃从而节省空间

      离屏渲染需要在屏幕外开辟内存空间,提前使用CPU渲染复杂的视图,保证视频控制器能够及时地从缓存区读到新的渲染结果。它在 GPU面临性能瓶颈时,将压力转移一部分给比较空闲的 CPU,然而 CPU 的渲染能力远没有 GPU 高效,有点杀鸡出牛刀的意思。同时这也是一种以空间换取时间的策略。离屏渲染的整个过程,需要多次切换上下文环境;等到离屏渲染结束以后,将离屏缓冲区的渲染结果显示到屏幕上有需要将上下文环境从离屏切换到当前屏幕。而上下文环境的切换是要付出很大代价的。

  • 如何检测离屏渲染

  1. 模拟器
    模拟器在工作栏上面的Debug -> Color Off-Screen Rendered
  2. 真机
    真机在工作栏上面的Debug -> View Debugging -> Rendering -> Color Off-Screen Rendered Yellow
    代码如下:
//1.按钮存在背景图片
    UIButton *btn1 = [UIButton buttonWithType:UIButtonTypeCustom];
    btn1.frame = CGRectMake(100, 30, 100, 100);
    btn1.layer.cornerRadius = 50;
    [self.view addSubview:btn1];
    
    [btn1 setImage:[UIImage imageNamed:@"btn.png"] forState:UIControlStateNormal];
    btn1.clipsToBounds = YES;
    
    //2.按钮不存在背景图片
    UIButton *btn2 = [UIButton buttonWithType:UIButtonTypeCustom];
    btn2.frame = CGRectMake(100, 180, 100, 100);
    btn2.layer.cornerRadius = 50;
    btn2.backgroundColor = [UIColor blueColor];
    [self.view addSubview:btn2];
    btn2.clipsToBounds = YES;
    
    //3.UIImageView 设置了图片+背景色;
    UIImageView *img1 = [[UIImageView alloc]init];
    img1.frame = CGRectMake(100, 320, 100, 100);
    img1.backgroundColor = [UIColor blueColor];
    [self.view addSubview:img1];
    img1.layer.cornerRadius = 50;
    img1.layer.masksToBounds = YES;
    img1.image = [UIImage imageNamed:@"btn.png"];
    
    //4.UIImageView 只设置了图片,无背景色;
    UIImageView *img2 = [[UIImageView alloc]init];
    img2.frame = CGRectMake(100, 480, 100, 100);
    [self.view addSubview:img2];
    img2.layer.cornerRadius = 50;
    img2.layer.masksToBounds = YES;
    img2.image = [UIImage imageNamed:@"btn.png"];

检测出现的离屏渲染的结果如下:


image.png

首先我们可以从代码中得出结论:设置圆角并不一定会触发离屏渲染。那什么样的情况下会触发离屏渲染呢?下面探讨一下


图片来自CC老师.png
  • 由图可以看出视图的层次结构:视图由背景色backgroundColor、内容contents、边缘borderWidth&borderColor构成

    image.png

通过官方文档解释可知设置layer.cornerRadius只会设置backgroundColorborder圆角,不会设置content圆角;除非同时设置了layer.masksToBoundstrue(对应的viewclipsToBounds属性)。此时两个属性相结合,产生离屏渲染。这也就说明了上面代码为什么1和3触发了离屏渲染,而2和4没有触发离屏渲染
由于视图是由多个图层组合的,当进行圆角操作是会触发离屏渲染。
应用cornerRadius和maskToBounds进行圆角+裁剪触发离屏渲染。

  • 离屏渲染什么时候被触发

    • 使用了mask的layer(mask.layer)
    • 需要进行裁剪的layer(layer.maskToBounds/View.clipsToBounds)
    • 设置了组透明度为YES,并且透明度不为1的layer(layer.allowsGroupOpacity/layer.opacity)
    • 添加了投影的layer(layer.shadow)
    • 采用了光栅化的layer(layer.shouldRasterize)
    • 绘制了文字的layer(UILable、CATextLayer、CoreText等)
  • 光栅化使用建议

    1. 如果layer不能被复用,则没有必要打开光栅化。

    2. 如果layer不是静态的,需要被频繁修改,比如处于动画之中,那么开启离屏渲染反而影响效率了。

    3. 离屏渲染缓存内容有时间限制,缓存内容100ms内容如果没有被使用,那么就会被丢弃;无法进行复用了。

    4. 离屏渲染缓存空间有限,超过2.5倍屏幕像素大小的话,也会失效且无法进行复用了。

  • 离屏渲染解决办法

    1. UI绘制圆角图片,前台进行设置

    2. 对于 contents 无内容或者内容的背景透明(无涉及到圆角以外的区域)的layer,直接设置layer的 backgroundColor 和 cornerRadius 属性来绘制圆角。

    3. 使用混合图层,在layer上方叠加相应mask形状的半透明layer
      sublayer.contents=(id)[UIImage imageNamed:@"xxx"].CGImage;
      [view.layer addSublayer:sublayer];

    4. -(UIImage *)yy_imageByRoundCornerRadius:(CGFloat)radius corners:(UIRectCorner)corners borderWidth:(CGFloat)borderWidth borderColor:(UIColor *)borderColor borderLineJoin:(CGLineJoin)borderLineJoin此方法为YY_image处理圆角的方法,你可以去下载YY_image查看源码

  • OpenGL渲染架构图分析

OpenGL渲染架构:
转自CC老师.png

上面的管线分成了客户端服务器端。
客户端是存储在CPU存储器中的,并在应用程序或者在内存的驱动程序中执行。驱动程序将渲染命令和数据组合起来,发送给服务器执行。
客户端不断的将数据块和命令组合起来送入缓冲区,缓存区会发送到 服务器端执行。服务器端将执行这些缓冲区的内容,客户端又做好了下一次发送渲染数据和命令的准备。

  • 着色器提供数据的三种方式
  1. attributes属性:用来传递顶点数据,投影矩阵,模型矩阵,只能传递到顶点着色器。
  2. Uniform值:统一的批次类,Uniform最常见的应用是在顶点渲染中设置变换矩阵,顶点着色器和片段着色器都可以有Uniform变量
  3. Texture纹理数据:纹理坐标给到顶点着色器,桥接给片元着色器。也可以直接给到片元着色器
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,793评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,567评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,342评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,825评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,814评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,680评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,033评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,687评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,175评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,668评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,775评论 1 332
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,419评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,020评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,978评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,206评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,092评论 2 351
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,510评论 2 343