jjzjj

第二十一篇:离屏渲染

坚持才会看到希望 2023-09-21 原文

我们经常用的UIKit框架就是继承与CoreAnimation,CoreGraphics框架。这两个框架又依赖于OpenGL ES。CoreImage是处理图像之前的一些操作。


WechatIMG2363.jpeg
WechatIMG2364.jpeg

下面这张图是核心:
通过视频控制器去frameBuffer也就是帧缓存里读取,就是通过下面的电子枪帧扫描读取。当扫描一圈后,也就是回到初始位置的时候,就会形成一个Vsync垂直信号。


WechatIMG2365.jpeg
WechatIMG2366.jpeg

离屏渲染的定义:

如果要在显示屏上显示内容,我们至少需要一块与屏幕像素数据量一样大
的frame buffer(帧缓冲区),作为像素数据存储区域,然后由显示控制器把帧缓存区的数据显示到屏幕上。如果有时因为面临一些限制,一些原因,比如说阴影,遮罩mask等,GPU无法把渲染结果直接写入frame buffer,而是先暂把中间的一个临时状态存在另外的内存区域,之后再写入frame buffer,那么这个过程被称之为离屏渲染。

UIview是基于CALayer的,CALayer其有三个部分,分别是backgroundColor,contents,borderWidthborderColor。其中clip to bounds属性就是设置修改contents的。

image.png
image.png

下面就是离屏渲染的过程,当要显示最后面一幅画时候,GPU在绘制时候数据时候,会先绘制个山,然后会绘制个草地,然后再绘制个树,然后会开辟了单独空间进行合成,这个空间也就是framebuffer。离屏渲染就是由于硬件的瓶颈限制导致的。


WechatIMG2379.jpeg

触发离屏渲染操作

光栅化一定会触发离屏渲染:

光栅化是一个缓存机制,开启的话就会以bitmap位图的形式保存起来,下次再使用的话就会从缓存中拿取进行渲染,可以减少CPU计算,其只能缓存100ms,所以很少用,其会触发离屏渲染。

  • (void)ShouldRasterize {
    self.lgImageView.layer.shouldRasterize = NO;
    }
遮罩mask一定会触发离屏渲染:

当添加遮罩,这个遮罩会放到CALayer的上层,这个就一定会导致离屏渲染。

  • (void)Mask {

    //添加到layer的上层
    CALayer *layer = [CALayer layer];
    layer.frame = CGRectMake(30, 30, self.lgImageView.bounds.size.width, self.lgImageView.bounds.size.height);
    layer.backgroundColor = [UIColor redColor].CGColor;
    self.lgImageView.layer.mask = layer;
    }

阴影一定会触发离屏渲染:

阴影是添加到layer的下层,层级就会比较复杂,就会触发离屏渲染。

  • (void)Shadows {

    //添加到layer的下层
    self.lgImageView.layer.shadowColor = [UIColor redColor].CGColor;
    self.lgImageView.layer.shadowOffset = CGSizeMake(20, 20);
    self.lgImageView.layer.shadowOpacity = 0.2;
    self.lgImageView.layer.shadowRadius = 5;
    self.lgImageView.layer.masksToBounds = NO;
    }

上面的阴影是可以优化的,我们可以提前指定阴影的路径,通过UIBezierPath曲线途径这样就不会触发离屏渲染了。

//阴影优化

  • (void)Shadows2 {
    self.lgImageView.layer.shadowColor = [UIColor redColor].CGColor;
    self.lgImageView.layer.shadowOpacity = 0.2;
    self.lgImageView.layer.shadowRadius = 5;
    self.lgImageView.layer.masksToBounds = NO;

    //提前指定阴影的路径
    [self.lgImageView.layer setShadowPath:[UIBezierPath bezierPathWithRect:CGRectMake(0, 0, self.lgImageView.bounds.size.width + 20, self.lgImageView.bounds.size.height + 20)].CGPath];
    }

抗锯齿在满足一定条件会触发离屏渲染。如果在设置属性时候,如果选择的模式为fit就会触发离屏渲染。选择的是fill就不会触发离屏渲染,是因为这样就不会有抗锯齿了,也就不会触发离屏渲染。

抗锯齿计算量也很大,对性能有要求

  • (void) EdgeAnntialiasing {
    CGFloat angle = M_PI / 60.0;
    [self.lgImageView.layer setTransform:CATransform3DRotate(self.lgImageView.layer.transform, angle, 0.0, 0.0, 1.0)];
    self.lgImageView.layer.allowsEdgeAntialiasing = YES;
    }
不透明不一定会触发离屏渲染(需要看起是否有子视图)。

allowsGroupOpacity 这个是设置了视图的子视图在透明度上是否和俯视图一致,可以看标注1,如果没有标注1这块是不会触发离屏渲染的。

  • (void)lgAllowsGroupOpacity {

    //标注1
    UIView *view = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 20, 20)];
    view.backgroundColor = [UIColor greenColor];
    view.alpha = 0.8;
    [self.lgImageView addSubview:view];
    //标注1

    self.lgImageView.alpha = 1;
    //allowsGroupOpacity 设置视图的子视图在透明度上是否和俯视图一致
    self.lgImageView.layer.allowsGroupOpacity = YES;
    }

当view上还有其它subview时
view 上子视图的 alpha 在 0 ~ 1 之间
view.layer.allowsGroupOpcity = Yes
具体原因在于重叠部分的图像需要两个view的图像合成来计算,因此只有满足上述三个条件时,才会触发。

圆角不一定会触发离屏渲染(label就不会触发离屏渲染,其他控件就会触发)。这个是因为lable我们在设置backgroundColor 颜色时候,其实是设置contents的颜色。
  • (void)lgRadius {

    //uilabel 设置圆角
    self.lgImageView.backgroundColor = [UIColor redColor];
    [self lgBezier];
    // self.lgImageView.layer.backgroundColor = [UIColor greenColor].CGColor;

    //contents + backgroundColor\border
    // self.lgImageView.layer.cornerRadius = 40;
    //
    //// self.lgLabel.backgroundColor = [UIColor redColor];
    // self.lgLabel.layer.backgroundColor = [UIColor greenColor].CGColor;
    // self.lgLabel.layer.cornerRadius = 10;
    }

在开发中尽量避免离屏渲染,因为会导致丢帧(卡顿),和消耗内存(因为其需要单独分配一个空间出来存framebuffer)。

drawRect方法讲解(一种特殊的离屏渲染):

使用drawRect方法会再开辟一块画布, 是在开辟的画布上画,在开发中尽量少重写drawRect方法。这个会增大内存消耗,这个被称为一种特殊的离屏渲染。操作drawRect后生成一个backing store,这个是村bitmap的,同时 会操作CoreGraphics框架,在CoreGraphics框架上操作这个是直接在CPU处理的。

有关第二十一篇:离屏渲染的更多相关文章

  1. ruby-on-rails - 渲染另一个 Controller 的 View - 2

    我想要做的是有2个不同的Controller,client和test_client。客户端Controller已经构建,我想创建一个test_clientController,我可以使用它来玩弄客户端的UI并根据需要进行调整。我主要是想绕过我在客户端中内置的验证及其对加载数据的管理Controller的依赖。所以我希望test_clientController加载示例数据集,然后呈现客户端Controller的索引View,以便我可以调整客户端UI。就是这样。我在test_clients索引方法中试过这个:classTestClientdefindexrender:template=>

  2. ruby-on-rails - Rails HTML 请求渲染 JSON - 2

    在我的Controller中,我通过以下方式在我的index方法中支持HTML和JSON:respond_todo|format|format.htmlformat.json{renderjson:@user}end在浏览器中拉起它时,它会自然地以HTML呈现。但是,当我对/user资源进行内容类型为application/json的curl调用时(因为它是索引方法),我仍然将HTML作为响应。如何获取JSON作为响应?我还需要说明什么? 最佳答案 您应该将.json附加到请求的url,提供的格式在routes.rb的路径中定义。这

  3. 世界前沿3D开发引擎HOOPS全面讲解——集3D数据读取、3D图形渲染、3D数据发布于一体的全新3D应用开发工具 - 2

    无论您是想搭建桌面端、WEB端或者移动端APP应用,HOOPSPlatform组件都可以为您提供弹性的3D集成架构,同时,由工业领域3D技术专家组成的HOOPS技术团队也能为您提供技术支持服务。如果您的客户期望有一种在多个平台(桌面/WEB/APP,而且某些客户端是“瘦”客户端)快速、方便地将数据接入到3D应用系统的解决方案,并且当访问数据时,在各个平台上的性能和用户体验保持一致,HOOPSPlatform将帮助您完成。利用HOOPSPlatform,您可以开发在任何环境下的3D基础应用架构。HOOPSPlatform可以帮您打造3D创新型产品,HOOPSSDK包含的技术有:快速且准确的CAD

  4. ruby - 如何跳过 CSV 文件的第一行并将第二行作为标题 - 2

    有没有办法跳过CSV文件的第一行,让第二行作为标题?我有一个CSV文件,第一行是日期,第二行是标题,所以我需要能够在遍历它时跳过第一行。我尝试使用slice但它会将CSV转换为数组,我真的很想将其读取为CSV,以便我可以利用header。 最佳答案 根据您的数据,您可以使用另一种方法和skip_lines-option此示例跳过所有以#开头的行require'csv'CSV.parse(DATA.read,:col_sep=>';',:headers=>true,:skip_lines=>/^#/#Markcomments!)do|

  5. ruby-on-rails - Rails 渲染带有驼峰命名法的 json 对象 - 2

    我在一个简单的RailsAPI中有以下Controller代码:classApi::V1::AccountsControllerehead:not_foundendendend问题在于,生成的json具有以下格式:{id:2,name:'Simpleaccount',cash_flows:[{id:1,amount:34.3,description:'simpledescription'},{id:2,amount:1.12,description:'otherdescription'}]}我需要我生成的json是camelCase('cashFlows'而不是'cash_flows'

  6. ruby-on-rails - 使用 header 渲染 JSON - 2

    我想在我的Controller中使用以下corsheader呈现JSON:'Access-Control-Allow-Origin'='*'.我试过这个:defmy_actionrender(json:some_params)response.headers['Access-Control-Allow-Origin']='*'end但是我得到了一个AbstractController::DoubleRenderError。有没有办法使用header呈现JSON? 最佳答案 您不能在渲染后设置header,因为已发送响应。所以在没有意

  7. ruby-on-rails - Ruby 数组到 JSON 和 Rails JSON 渲染 - 2

    我有一个Ruby数组,如何在Rails3.0中将其呈现为JSONView?我的Controller方法是defautocomplete@question=Question.allend 最佳答案 如果自动完成操作仅呈现JSON,您可以将re5et的解决方案简化为:defautocompletequestions=Question.allrender:json=>questionsend(请注意,我将“问题”复数化以反射(reflect)它是一个数组并删除了@符号-一个局部变量就足够了,因为您可能只使用它来呈现内联JSON)作为一种附

  8. ruby - `respond_to_missing?` 的第二个参数有什么用吗? - 2

    使用method_missing时在Ruby中,它是almostalwaysagoodidea定义respond_to_missing?respond_to_missing?接受两个参数;我们正在检查的方法的名称(symbol),以及一个指示我们是否应该在检查中包含私有(private)方法的bool值(include_all)。现在我感到困惑的是:method_missing不接受任何可能指示它是否应该调用私有(private)方法的参数,如respond_to_missing?做。此外,method_missing无论原始方法调用是在公共(public)上下文还是私有(privat

  9. ruby-on-rails - 使用 ApplicationController.renderer.render 从 Controller 外部渲染的 Rails 5 不会在自身上设置变量 - 2

    我正在使用Rails5ApplicationController.renderer.render方法从模型中进行渲染。我需要将一些变量传递给我的布局,这是我使用locals选项完成的;如果直接访问此变量,则该变量在布局中可用,但不能通过self访问。这是我设置渲染的方式html_string=ApplicationController.renderer.render(file:"/#{template_path}/base/show",:formats=>[:pdf,:html],locals:{:@routing_form=>self,:controller_name=>contro

  10. ruby-on-rails - 为什么我的 Rails 服务器渲染时间不加起来? - 2

    我的Rails应用程序在暂存服务器上运行速度非常慢,这让我遇到了一些麻烦。最令人困惑的是每个请求的日志输出的最后一行。看起来View和数据库时间甚至不接近整个渲染时间。在一页上,完成时间大约1000毫秒,View大约450毫秒,数据库大约20毫秒。渲染页面所需的其余时间从何而来? 最佳答案 当事情变得神秘时......分析器是你的friend!分析器将统计哪些方法被调用最多以及每个方法调用花费多长时间。ruby-prof当我在RubyLand时,它会帮我解决这个问题,它会生成一个漂亮的调用图(如果需要,可以是html格式),这使得查

随机推荐