草庐IT

【Unity TIL】Unity 通过 DrawInstance 一次绘制多个实例(三)

fcnaud 2023-03-28 原文

游戏开发中,可能会碰到一次绘制多个拥有相同 mesh 的物体,它们可能是位置,旋转等不同,或者是材质的某些参数不同,我们要做的就是配合 Unity 的渲染方式,尽可能地减少绘制的操作。

由于下文主要讨论的是 DrawInstance 和 MaterialPropertyBlock,所以其他的一下影响合批的内容没有讨论。

动态合批

绘制多个相同 Mesh,相同 Material 的物体时,打开 Material 的 Gpu Instance 选项,在满足顶点限制的条件下,会进行动态合批(动态合批需要打开)。

不同 Material 会打断合批

修改 Material 的参数,这样会打断合批。并且 Unity 会为每一个物体创建一个 Material 的实例。

public class TestDrawInstance : MonoBehaviour
{
    public Transform[] Cubes;
    
    void Start()
    {
        foreach (var cube in Cubes)
        {
            cube.GetComponent<Renderer>().material.
                SetColor("_Color", new Color(
                Random.Range(0f, 1f),
                Random.Range(0f, 1f),
                Random.Range(0f, 1f)));
        }
    }
}

通过 MaterialPropertyBlock 避免创建 Material 实例

这时可以借助 MaterialPropertyBlock 来设置不同的属性,但是这只会节省,创建多个 Material 的消耗,并不会影响到渲染,依然会打断合批。 不会打断,满足动态合批的条件时,会进行合批。

MaterialPropertyBlock propertyBlock = new MaterialPropertyBlock();
foreach (var cube in Cubes)
{
    // 设置不同的属性
    propertyBlock.SetColor("_Color", new Color(
        Random.Range(0f, 1f),
        Random.Range(0f, 1f),
        Random.Range(0f, 1f)));
    
    // 设置 PropertyBlock 替代设置 material 属性
    cube.GetComponent<Renderer>()
        .SetPropertyBlock(propertyBlock);
}

DrawMeshInstanced 一次性绘制

直接使用 DrawMeshInstanced API 也可以达到合批的作用,但是无法设置不同的参数。同时需要注意一次 DrawInstance 调用最多绘制 1023 个物体的限制。

public Transform[] Cubes;
private Material drawMat;
Mesh drawMesh;
Matrix4x4[] trsMat;

private void Update()
{
    // 使用 DrawInstance 绘制一次性绘制
    drawMat = Cubes[0].GetComponent<Renderer>().sharedMaterial;
    drawMesh = Cubes[0].GetComponent<MeshFilter>().sharedMesh;
    trsMat = new Matrix4x4[Cubes.Length];

    for (int i = 0; i < Cubes.Length; i++) {
        trsMat[i] = Matrix4x4.TRS(
            Cubes[i].position, 
            Cubes[i].rotation, 
            Cubes[i].lossyScale);
        Cubes[i].GetComponent<Renderer>().enabled = false;
    }
    
    Graphics.DrawMeshInstanced(
        drawMesh,
        0,
        drawMat,
        trsMat,
        Cubes.Length,
        null,
        ShadowCastingMode.Off,
        false
        );
}

DrawMeshInstanced 配合 MaterialPropertyBlock 进行绘制

DrawMeshInstanced 和 MaterialPropertyBlock 同样是可以进行配合的,但是需要编写能处理 MaterialPropertyBlock 数据的 shader。

首先要 编写支持 GPU Instancing 的 Shader。以顶点片元 shader 为例,需要如下步骤

  1. 开启 gpu instance。 #pragma multi_compile_instancing
  2. 在结构体中添加 id 声明。UNITY_VERTEX_INPUT_INSTANCE_ID
  3. 定义数据。
  4. setup,在 vert/frag 中使用需要先 UNITY_SETUP_INSTANCE_ID。在 frag 中使用还要在 vert 中使用 UNITY_TRANSFER_INSTANCE_ID
  5. 获取数据。UNITY_ACCESS_INSTANCED_PROP

shader sample


Shader "Custom/TestDrawInstance"
{
    Properties
    {
        _MainTex ("Texture", 2D) = "white" {}
    }
    SubShader
    {
        Tags { "RenderType"="Opaque" }
        LOD 100

        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            // make fog work
            #pragma multi_compile_fog
            // 1. 开启 gpu instancing
            #pragma multi_compile_instancing

            #include "UnityCG.cginc"

            struct appdata
            {
                float4 vertex : POSITION;
                float2 uv : TEXCOORD0;
                // 2. 在数据中添加一个语义为 SV_InstanceID 的元素
                // 在 vertex 中使用
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            struct v2f
            {
                float2 uv : TEXCOORD0;
                UNITY_FOG_COORDS(1)
                float4 vertex : SV_POSITION;
                
                // 2. 在数据中添加一个语义为 SV_InstanceID 的元素
                // 在 fragment 中使用
                UNITY_VERTEX_INPUT_INSTANCE_ID
            };

            // 3. 定义数据, name 为任意字符串, START 和 END 成对使用
            UNITY_INSTANCING_BUFFER_START(Props)
                // 定义一个实例属性
                UNITY_DEFINE_INSTANCED_PROP(float4, _Color)
            UNITY_INSTANCING_BUFFER_END(Props)

            sampler2D _MainTex;
            float4 _MainTex_ST;

            v2f vert (appdata v)
            {
                v2f o;
                
                // 4. setup
                UNITY_SETUP_INSTANCE_ID(v);
                // 在 fragment 中使用,需要在此进行设置
                UNITY_TRANSFER_INSTANCE_ID(v, o);
                
                o.vertex = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                UNITY_TRANSFER_FOG(o,o.vertex);
                return o;
            }

            fixed4 frag (v2f i) : SV_Target
            {
                // 4. setup
                UNITY_SETUP_INSTANCE_ID(i);
                // sample the texture
                fixed4 col = tex2D(_MainTex, i.uv);
                // 5. 获取实例数据
                col = UNITY_ACCESS_INSTANCED_PROP(Props, _Color);
                // apply fog
                UNITY_APPLY_FOG(i.fogCoord, col);
                return col;
            }
            ENDCG
        }
    }
}

使用代码控制绘制

public class TestDrawInstance : MonoBehaviour
{
    public Transform[] Cubes;
    private Material drawMat;
    private Mesh drawMesh;
    private Matrix4x4[] trsMat;
    private MaterialPropertyBlock instanceBlock;
    private Vector4[] colors;

    private void Start()
    {
        drawMat = Cubes[0].GetComponent<Renderer>().sharedMaterial;
        drawMesh = Cubes[0].GetComponent<MeshFilter>().sharedMesh;
        trsMat = new Matrix4x4[Cubes.Length];
        colors = new Vector4[Cubes.Length];

        instanceBlock = new MaterialPropertyBlock();
        for (int i = 0; i < Cubes.Length; i++)
        {
            colors[i] = new Vector4(
                Random.Range(0f, 1f),
                Random.Range(0f, 1f),
                Random.Range(0f, 1f),
                1);
            Cubes[i].GetComponent<Renderer>().enabled = false;
        }
        instanceBlock.SetVectorArray("_Color", colors);
    }

    private void Update()
    {
        for (int i = 0; i < Cubes.Length; i++) {
            trsMat[i] = Matrix4x4.TRS(
                Cubes[i].position, 
                Cubes[i].rotation, 
                Cubes[i].lossyScale);
        }
        
        Graphics.DrawMeshInstanced(
            drawMesh,
            0,
            drawMat,
            trsMat,
            Cubes.Length,
            instanceBlock,
            ShadowCastingMode.Off,
            false
            );
    }
}

参考链接

有关【Unity TIL】Unity 通过 DrawInstance 一次绘制多个实例(三)的更多相关文章

  1. ruby-on-rails - Rails 3 中的多个路由文件 - 2

    Rails2.3可以选择随时使用RouteSet#add_configuration_file添加更多路由。是否可以在Rails3项目中做同样的事情? 最佳答案 在config/application.rb中:config.paths.config.routes在Rails3.2(也可能是Rails3.1)中,使用:config.paths["config/routes"] 关于ruby-on-rails-Rails3中的多个路由文件,我们在StackOverflow上找到一个类似的问题

  2. ruby-on-rails - 在 Ruby 中循环遍历多个数组 - 2

    我有多个ActiveRecord子类Item的实例数组,我需要根据最早的事件循环打印。在这种情况下,我需要打印付款和维护日期,如下所示:ItemAmaintenancerequiredin5daysItemBpaymentrequiredin6daysItemApaymentrequiredin7daysItemBmaintenancerequiredin8days我目前有两个查询,用于查找maintenance和payment项目(非排他性查询),并输出如下内容:paymentrequiredin...maintenancerequiredin...有什么方法可以改善上述(丑陋的)代

  3. ruby - 通过 rvm 升级 ruby​​gems 的问题 - 2

    尝试通过RVM将RubyGems升级到版本1.8.10并出现此错误:$rvmrubygemslatestRemovingoldRubygemsfiles...Installingrubygems-1.8.10forruby-1.9.2-p180...ERROR:Errorrunning'GEM_PATH="/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/ruby-1.9.2-p180@global:/Users/foo/.rvm/gems/ruby-1.9.2-p180:/Users/foo/.rvm/gems/rub

  4. ruby - 使用 Vim Rails,您可以创建一个新的迁移文件并一次性打开它吗? - 2

    使用带有Rails插件的vim,您可以创建一个迁移文件,然后一次性打开该文件吗?textmate也可以这样吗? 最佳答案 你可以使用rails.vim然后做类似的事情::Rgeneratemigratonadd_foo_to_bar插件将打开迁移生成的文件,这正是您想要的。我不能代表textmate。 关于ruby-使用VimRails,您可以创建一个新的迁移文件并一次性打开它吗?,我们在StackOverflow上找到一个类似的问题: https://sta

  5. ruby-on-rails - Rails - 一个 View 中的多个模型 - 2

    我需要从一个View访问多个模型。以前,我的links_controller仅用于提供以不同方式排序的链接资源。现在我想包括一个部分(我假设)显示按分数排序的顶级用户(@users=User.all.sort_by(&:score))我知道我可以将此代码插入每个链接操作并从View访问它,但这似乎不是“ruby方式”,我将需要在不久的将来访问更多模型。这可能会变得很脏,是否有针对这种情况的任何技术?注意事项:我认为我的应用程序正朝着单一格式和动态页面内容的方向发展,本质上是一个典型的网络应用程序。我知道before_filter但考虑到我希望应用程序进入的方向,这似乎很麻烦。最终从任何

  6. ruby - 如何每月在 Heroku 运行一次 Scheduler 插件? - 2

    在选择我想要运行操作的频率时,唯一的选项是“每天”、“每小时”和“每10分钟”。谢谢!我想为我的Rails3.1应用程序运行调度程序。 最佳答案 这不是一个优雅的解决方案,但您可以安排它每天运行,并在实际开始工作之前检查日期是否为当月的第一天。 关于ruby-如何每月在Heroku运行一次Scheduler插件?,我们在StackOverflow上找到一个类似的问题: https://stackoverflow.com/questions/8692687/

  7. ruby - 通过 erb 模板输出 ruby​​ 数组 - 2

    我正在使用puppet为ruby​​程序提供一组常量。我需要提供一组主机名,我的程序将对其进行迭代。在我之前使用的bash脚本中,我只是将它作为一个puppet变量hosts=>"host1,host2"我将其提供给bash脚本作为HOSTS=显然这对ruby​​不太适用——我需要它的格式hosts=["host1","host2"]自从phosts和putsmy_array.inspect提供输出["host1","host2"]我希望使用其中之一。不幸的是,我终其一生都无法弄清楚如何让它发挥作用。我尝试了以下各项:我发现某处他们指出我需要在函数调用前放置“function_”……这

  8. ruby-on-rails - 如何使用 instance_variable_set 正确设置实例变量? - 2

    我正在查看instance_variable_set的文档并看到给出的示例代码是这样做的:obj.instance_variable_set(:@instnc_var,"valuefortheinstancevariable")然后允许您在类的任何实例方法中以@instnc_var的形式访问该变量。我想知道为什么在@instnc_var之前需要一个冒号:。冒号有什么作用? 最佳答案 我的第一直觉是告诉你不要使用instance_variable_set除非你真的知道你用它做什么。它本质上是一种元编程工具或绕过实例变量可见性的黑客攻击

  9. ruby 正则表达式 - 如何替换字符串中匹配项的第 n 个实例 - 2

    在我的应用程序中,我需要能够找到所有数字子字符串,然后扫描每个子字符串,找到第一个匹配范围(例如5到15之间)的子字符串,并将该实例替换为另一个字符串“X”。我的测试字符串s="1foo100bar10gee1"我的初始模式是1个或多个数字的任何字符串,例如,re=Regexp.new(/\d+/)matches=s.scan(re)给出["1","100","10","1"]如果我想用“X”替换第N个匹配项,并且只替换第N个匹配项,我该怎么做?例如,如果我想替换第三个匹配项“10”(匹配项[2]),我不能只说s[matches[2]]="X"因为它做了两次替换“1fooX0barXg

  10. ruby - 通过 ruby​​ 进程共享变量 - 2

    我正在编写一个gem,我必须在其中fork两个启动两个webrick服务器的进程。我想通过基类的类方法启动这个服务器,因为应该只有这两个服务器在运行,而不是多个。在运行时,我想调用这两个服务器上的一些方法来更改变量。我的问题是,我无法通过基类的类方法访问fork的实例变量。此外,我不能在我的基类中使用线程,因为在幕后我正在使用另一个不是线程安全的库。所以我必须将每个服务器派生到它自己的进程。我用类变量试过了,比如@@server。但是当我试图通过基类访问这个变量时,它是nil。我读到在Ruby中不可能在分支之间共享类变量,对吗?那么,还有其他解决办法吗?我考虑过使用单例,但我不确定这是

随机推荐