unity的CommandBuffer介绍

news2025/1/11 2:32:25

  大家好,我是阿赵。
  之前介绍过使用PostProcessing来做屏幕后处理效果。我们不一定要用PostProcessing来做后处理效果。
  PostProcessing功能强大,比如不同的layer控制不同的屏幕效果,比如可以使用PostProcessVolume的非全局效果达到某个范围内有过渡性的后处理效果。但如果我们并没有用到这些效果,只是单纯的想在自己需要的时候,添加一个指定摄像机的屏幕效果,那么其实选择还有很多,比如Unity本身提供了Graphics图形接口,还有CommandBuffer相关的方法,也是可以直接制作屏幕效果的。
  从PostProcessing的实现原理来说,他其实也是使用了CommandBuffer来实现,只是对它进行了封装。
  下面来介绍一下CommandBuffer。

一、Graphics和CommandBuffer

  在PostProcessing出现之前,我们同样可以制作屏幕后处理效果,一般的做法是:
  在OnRenderImage声明周期方法里面,通过Graphics.Blit(source, destination, material)方法,对摄像机传入的屏幕渲染结果,通过自己指定的材质球里面的shader处理,得到一张新的屏幕渲染效果,然后显示在屏幕上面。
  这里出现了Graphics。
  Graphics是Unity提供的图形绘制接口,除了Blit方法,还有很多其他的绘制方法,比如CopyTexture、DrawMesh、DrawTexture等。
  然后看CommandBuffer,它也提供了CommandBuffer.Blit方法,也提供了DrawMesh之类的方法。其实很多人到了这一步就开始有点困惑了。Graphics和CommandBuffer有什么区别呢?为什么都提供差不多的方法呢?什么时候该用Graphics,什么时候该用CommandBuffer呢?
  虽然两者很相似,但他们两个代表的含义是不一样的。

1、Graphics

  当我们调用Graphics方法时,它都是直接生效的,或者下一帧就生效的。比如我们调用Graphics.Blit方法,它会立刻就把原图通过shader处理,输出一张新图。比如我们DrawMesh,它会在下一帧就把我们需要的网格模型渲染出来。如果用DrawMeshNow,它就会在同一帧立刻把网格模型渲染出来。它是一个实际执行的命令。

2、CommandBuffer

  CommandBuffer是一个对象,其实是一个命令的列表。我们可以创建一个CommandBuffer对象,然后把需要它做的事情添加到这个对象的命令列表里面。至于什么时候执行,是可以控制的。比如我们可以把这个列表命令添加到摄像机渲染的某一个过程中间,或者灯光渲染的某个过程中间执行。也可以调用Graphics.ExecuteCommandBuffer方法,立刻执行这些命令。

二、CommandBuffer的执行过程

1、正常的渲染过程

  如果我们打开Unity自带的FrameDebug工具,可以看到整个渲染的过程
比如在Deferred延迟渲染里面,过程如下图:
在这里插入图片描述

如果是Forward前向渲染,过程会如下图
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

  如果选中其中的步骤,可以看到该步骤是渲染了什么画面,一步一步的看这个过程,对我们理解渲染是很有帮助的。

2、插入CommandBuffer的过程

  通过添加CommandBuffer命令,我们可以做到的是,在上面的渲染过程的其中一个步骤,插入自己想要的渲染处理。
  比如我这里截取一段代码:

        cmd1 = new CommandBuffer();
        cmd1.name = "AzhaoDrawLightObj";

        cmd1.SetRenderTarget(rt1);
        cmd1.ClearRenderTarget(true, true, Color.clear);

        for (int i = 0; i < renders.Length; i++)
        {
            Renderer item = renders[i];
            if (item.gameObject.activeInHierarchy == false || item.enabled == false)
            {
                continue;
            }
            if(item.gameObject.layer == 8)
            {
                cmd1.DrawRenderer(item, whiteMat);
            }
            else
            {
                cmd1.DrawRenderer(item, blackMat);
            }
            
        }

        cmd1.Blit(rt1, rt2);
        for (int i = 0; i < iterations; i++)
        {
            blurMat.SetFloat("_BlurSize", 1.0f + i * blurSpread);

            cmd1.Blit(rt2, rt3, blurMat, 0); 

            cmd1.Blit(rt3, rt2, blurMat, 1); 

        }
        cmd1.Blit(rt2, rt1);
        comMat.SetTexture("_AddTex", rt1);
        cam.AddCommandBuffer(CameraEvent.BeforeImageEffects, cmd1);

  这里我创建了一个名字叫做AzhaoDrawLightObj的CommandBuffer命令列表,然后我打算在刚才2个球的中间,渲染一个会发光的Cube立方体,所以给这个叫做AzhaoDrawLightObj的CommandBuffer添加了很多命令。具体这些命令是干什么用的,现在不需要去深究,之后我会单独写文章说明。
这里只需要关注两段:
第一段:

cmd1 = new CommandBuffer();
cmd1.name = "AzhaoDrawLightObj";
cmd1.SetRenderTarget(rt1);
cmd1.ClearRenderTarget(true, true, Color.clear);

  这一段是创建了CommandBuffer,给它命名,然后设置目标渲染的RenderTexture,并且清理这张RenderTexture
第二段:

cam.AddCommandBuffer(CameraEvent.BeforeImageEffects, cmd1);

  这一句代码,cam是一个Camera摄像机,通过AddCommandBuffer方法,我把刚刚创建的CommandBuffer添加到摄像机上面。CameraEvent.BeforeImageEffects是指定这个CommandBuffer生效的时机。
再回头看FrameDebug的渲染过程:
在这里插入图片描述

  会发现在渲染过程中,插入了BeforeImageEffects这个过程,里面就看到我刚刚添加的AzhaoDrawLightObj的过程了。
在这里插入图片描述

  现在在两个球中间,就单独出现了一个会放过的立方体了。使用CommandBuffer是比较自由的,可以控制渲染单个物体,可以控制在某个时机处理特定的渲染,这些自由度,是使用固定的PostProcessing很难做得到的。
在这里插入图片描述

  这时候可以在摄像机上面,看到有哪些CommandBuffer添加了。
  值得注意的是,给摄像机添加CommandBuffer,只需要在OnEnable生命周期添加一次,就能一直生效,不要在Update添加,不然会发现重复添加了非常多。
这是在Update里面调用添加的后果:
在这里插入图片描述

既然是在OnEnable里面添加的,也要记得在OnDisable里面删除。

void OnDisable()
{
    if (cmd1!=null)
    {
        cmd1.Dispose();
        cmd1 = null;
    }
}

三、添加CommandBuffer的事件说明

  这里说明的是CameraEvent事件,在不同的渲染模式下,使用的CameraEvent事件是不一样的,千万不要用错了。除了CameraEvent还有LightEvent
  下面是Unity的官方文档里面对CameraEvent和LightEvent执行顺序的说明

1、Deferred rendering path

CameraEvent.BeforeGBuffer
Unity renders opaque geometry
CameraEvent.AfterGBuffer
Unity resolves depth.
CameraEvent.BeforeReflections
Unity renders default reflections, and Reflection Probe reflections.
CameraEvent.AfterReflections
Unity copies reflections to the Emissive channel of the G-buffer.
CameraEvent.BeforeLighting
Unity renders shadows. See LightEvent order of execution.
CameraEvent.AfterLighting
CameraEvent.BeforeFinalPass
Unity processes the final pass.
CameraEvent.AfterFinalPass
CameraEvent.BeforeForwardOpaque (only called if there is opaque geometry that cannot be rendered using deferred)
Unity renders opaque geometry that cannot be rendered with deferred rendering.
CameraEvent.AfterForwardOpaque (only called if there is opaque geometry that cannot be rendered using deferred)
CameraEvent.BeforeSkybox
Unity renders the skybox
CameraEvent.AfterSkybox
Unity renders halos.
CameraEvent.BeforeImageEffectsOpaque
Unity applies opaque-only post-processing effects.
CameraEvent.AfterImageEffectsOpaque
CameraEvent.BeforeForwardAlpha
Unity renders transparent geometry, and UI Canvases with a Rendering Mode of Screen Space - Camera
.
CameraEvent.AfterForwardAlpha
CameraEvent.BeforeHaloAndLensFlares
Unity renders lens flares.
CameraEvent.AfterHaloAndLensFlares
CameraEvent.BeforeImageEffects
Unity applies post-processing effects.
CameraEvent.AfterImageEffects
CameraEvent.AfterEverything
Unity renders UI Canvases with a Rendering Mode that is not Screen Space - Camera.

2、Forward rendering path

CameraEvent.BeforeDepthTexture
Unity renders depth for opaque geometry.
CameraEvent.AfterDepthTexture
CameraEvent.BeforeDepthNormalsTexture
Unity renders depth normals for opaque geometry.
CameraEvent.AfterDepthNormalsTexture
Unity renders shadows. See LightEvent order of execution.
CameraEvent.BeforeForwardOpaque
Unity renders opaque geometry.
CameraEvent.AfterForwardOpaque
CameraEvent.BeforeSkybox
Unity renders the skybox.
CameraEvent.AfterSkybox
Unity renders halos.
CameraEvent.BeforeImageEffectsOpaque
Unity applies opaque-only post-processing effects.
CameraEvent.AfterImageEffectsOpaque
CameraEvent.BeforeForwardAlpha
Unity renders transparent geometry, and UI Canvases with a Rendering Mode of Screen Space - Camera.
CameraEvent.AfterForwardAlpha
CameraEvent.BeforeHaloAndLensFlares
Unity renders lens flares.
CameraEvent.AfterHaloAndLensFlares
CameraEvent.BeforeImageEffects
Unity applies post-processing effects.
CameraEvent.AfterImageEffects
CameraEvent.AfterEverything
Unity renders UI Canvases with a Rendering Mode that is not Screen Space - Camera.

3、LightEvent

LightEvent.BeforeShadowMap
LightEvent.BeforeShadowMapPass
Unity renders all shadow casters for the current Pass
LightEvent.AfterShadowMapPass
Unity repeats the last three steps, for each Pass
LightEvent.AfterShadowMap
LightEvent.BeforeScreenSpaceMask
Unity gathers the shadow map into a screen space buffer and performs filtering *AfterScreenSpaceMask

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/749540.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【三维重建】【深度学习】NeuS总览

【三维重建】【深度学习】NeuS总览 论文提出了一种新颖的神经表面重建方法&#xff0c;称为NeuS&#xff0c;用于从2D图像输入以高保真度重建对象和场景。在NeuS中建议将曲面表示为有符号距离函数(SDF)的零级集&#xff0c;并开发一种新的体绘制方法来训练神经SDF表示&#xff…

【Distributed】分布式ELK日志文件分析系统(一)

文章目录 一、ELK 概述1. 为什么要使用 ELK2. 完整日志系统基本特征3. ELK 简介3.1 ElasticSearch&#xff08;ES&#xff09;3.2 Kiabana3.3 Logstash3.4 其它组件Filebeat缓存/消息队列Fluentd 4. ELK 的工作原理5. Linux 系统内核日志消息的优先级别 二、 部署 ELK 集群服务…

二叉树题目:合并二叉树

文章目录 题目标题和出处难度题目描述要求示例数据范围 解法一思路和算法代码复杂度分析 解法二思路和算法代码复杂度分析 题目 标题和出处 标题&#xff1a;合并二叉树 出处&#xff1a;617. 合并二叉树 难度 3 级 题目描述 要求 给定两个二叉树 root1 \texttt{root1…

java项目之多人命题系统(ssm+mysql+jsp)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的多人命题系统。技术交流和部署相关看文章末尾&#xff01; 开发环境&#xff1a; 后端&#xff1a; 开发语言&#xff1a;Java 框架&…

zabbix部署及应用(自定义监控内容,zabbix 自动发现与自动注册,zabbix 代理服务器,Zabbix 高可用集群)

zabbix部署及应用 一、添加 zabbix 客户端主机1、服务端和客户端都配置时间同步2、服务端和客户端都设置 hosts 解析3、设置 zabbix 的下载源&#xff0c;安装 zabbix-agent24、修改 agent2 配置文件5、启动 zabbix-agent26、在服务端验证 zabbix-agent2 的连通性7、常用的键值…

一、对象的概念(1)

本章概要 抽象接口服务提供封装 抽象 “我们没有意识到惯用语言的结构有多大的力量。可以毫不夸张地说&#xff0c;它通过语义反应机制奴役我们。语言表现出来并在无意识中给我们留下深刻印象的结构会自动投射到我们周围的世界。” – Alfred Korzybski (1930) 计算机革命的起…

分布式锁【 基于synchronized锁解决超卖问题、分布式锁解决方案、悲观锁实现的分布式锁】(二)-全面详解(学习总结---从入门到深化)

目录 分布式锁问题_演示问题 基于synchronized锁解决超卖问题 分布式锁解决方案 分布式锁实现方案 分布式锁解决方案_数据库悲观锁实现的分布式锁 项目中使用for update 分布式锁问题_演示问题 启动订单服务9090 启动订单服务9091 创建两个SpringBoot服务 启动Nginx服务 下载N…

火山引擎云搜索服务升级云原生新架构;提供数十亿级分布式向量数据库能力

从互联网发展伊始&#xff0c;搜索技术就绽放出了惊人的社会和经济价值。随着信息社会快速发展&#xff0c;数据呈爆炸式增长&#xff0c;搜索技术通过数据收集与处理&#xff0c;满足信息共享与快速检索的需求。 云搜索服务 ESCloud 是火山引擎提供的完全托管在线分布式搜索服…

OJ练习第135题——下降路径最小和

下降路径最小和 力扣链接&#xff1a;931. 下降路径最小和 题目描述 给你一个 n x n 的 方形 整数数组 matrix &#xff0c;请你找出并返回通过 matrix 的下降路径 的 最小和 。 下降路径 可以从第一行中的任何元素开始&#xff0c;并从每一行中选择一个元素。在下一行选择…

永磁同步电机磁场定向控制(FOC)及Matlab/Simulink仿真分析

前言 做永磁同步电机控制绕不开FOC&#xff0c;本章节主要介绍FOC控制的基本原理、坐标变换以及永磁同步电机在同步旋转坐标系下的数学模型&#xff0c;并通过Matlab/Simulink进行永磁同步电机FOC控制算法的仿真分析。 01 FOC的基本原理 磁场定向控制&#xff08;Field-Ori…

分割3-转置卷积可逆吗?

上一话 分割2——语义分割评价指标https://blog.csdn.net/XiaoyYidiaodiao/article/details/125149509?spm1001.2014.3001.5502 转置卷积不是卷积的逆操作&#xff0c;并且转置卷积是卷积操作&#xff01; 这是因为普通卷积的操作图 1. 图1 其卷积可等效为图 2. 图2 其卷积操…

Offset Explorer2 监视kafka的利器

kafka作为一个生产者和消费者集为一体的框架&#xff0c;消费者必须一直保持打开的状态&#xff0c;并且每隔一段时间接收一次数据&#xff0c;才能够保持生产者放入的数据及时被处理掉&#xff0c;而生产者则可以每隔一段时间发送一波数据&#xff0c;这样消费者就能够接收到了…

Mysql搭建互为主从数据库

Mysql搭建互为主从数据库 一、搭建前期说明二、mysql文件结构以及配置说明1、mysqlA文件结构2、mysqlB文件结构3、mysqlA的配置文件my.cnf4、mysqlB的配置文件my.cnf5、启动mysqlA服务器脚本startMysqlA.sh6、启动mysqlB服务器脚本startMysqlB.sh7、查看服务启动情况 三、设置主…

【Redis】之缓存一致性

1、缓存一致性 对于使用 Redis 作为缓存来说&#xff0c;如何保证数据库和缓存数据一致性是个麻烦的问题。对于缓存和数据库的操作&#xff0c;主要有以下两种方式&#xff1a; 先删缓存&#xff0c;再更新数据库&#xff1b;先更新数据库&#xff0c;再删除缓存&#xff1b;…

【UE4 塔防游戏系列】02-基础设置

步骤 1. 新建一个蓝图&#xff0c;父类为游戏模式基础 命名为“TaFangGameMode” 2. 新建一个玩家控制器 命名为“TaFangGamePlayerController” 3. 在世界场景设置中&#xff0c;选择游戏覆盖模式为“TaFangGameMode” 默认Pawn类设为None&#xff0c;玩家控制器类选择“TaFa…

使用rest-assured框架优雅实现接口测试断言

对于接口测试来说&#xff0c;只判断响应状态码为200是远远不够的&#xff0c;我们经常必须去验证响应内容中的业务字段是否正确&#xff0c;本文将采用Rest-Assured框架来实现接口响应的判断。 1、接口信息的调试 当我们在做接口测试时&#xff0c;通常希望将接口的请求及响…

包装类(wrapper)

包装类(wrapper) 包装类和基本数据类型间的转换(Interger演示) public class WrapperType01 {public static void main(String[] args) {int a 10;// 手动封箱&#xff0c;Integer integer Integer.valueOf(a);System.out.println(integer);// 手动拆箱int i integer.intVa…

十五周算法训练营——岛屿问题

今天是十五周算法训练营的第十五周&#xff0c;主要讲岛屿问题专题。&#xff08;欢迎加入十五周算法训练营&#xff0c;与小伙伴一起卷算法&#xff09; 岛屿问题 一、题目 给你一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的的二维网格&#xff…

11. 利用Tomcat服务器配置HTTPS双向认定

文章目录 Tomcat配置HTTPS1.为服务器生成证书2.为客户端生成证书3.让服务器信任客户端证书4.将该文件导入到服务器的证书库&#xff0c;添加为一个信任证书使用命令如下&#xff1a;5.查看证书库6.让客户端信任服务器证书7.配置tomcat8.验证 Tomcat配置HTTPS 1.启动cmd控制台&…

分布式锁【分布式锁概述、业务介绍、创建SpringBoot项目】(一)-全面详解(学习总结---从入门到深化)

目录 分布式锁概述 分布式锁问题_业务介绍 分布式锁问题_创建SpringBoot项目 分布式锁概述 为什么需要分布式锁 在单机部署的系统中&#xff0c;使用线程锁来解决高并发的问题&#xff0c;多线程访问共享变量的问题达到数据一致性&#xff0c;如使用synchornized、 Reentran…