Unity中Shader特性PerRendererData

文章目录


前言

Unity中Shader特性PerRendererData


一、优化前是对使用了相同材质球的不同物体间shader分别设置,比较消耗性能

这里在shader中使用一个Color属性进行测试

复制代码
Shader"MyShader/P1_1_3"
{
    Properties
    {
        //命名要按标准来,这个属性才可以和Unity组件中的属性产生关联
        //比如说,在更改 Image 的源图片时,同时更改这个
        [PerRendererData]_MainTex("MainTex",2D) = "white"{}
        
        [PerRendererData]_Color("Color",color) = (1,1,1,1)
    }
    SubShader
    {
        //更改渲染队列(UI的渲染队列一般是半透明层的)
        Tags {"Queue" = "TransParent"}
        //混合模式
        Blend SrcAlpha OneMinusSrcAlpha
        Pass
        {
            CGPROGRAM
            #pragma vertex  vert
            #pragma fragment frag
            #include "UnityCG.cginc"
            //存储 应用程序输入到顶点着色器的信息
            struct appdata
            {
                //顶点信息
                float4 vertex:POSITION;

                float2 uv : TEXCOORD;
            };
            //存储 顶点着色器输入到片元着色器的信息
            struct v2f
            {
                //裁剪空间下的位置信息
                float4 pos:SV_POSITION;
                float2 uv : TEXCOORD;
            };
            
            sampler2D _MainTex;
            fixed4 _Color;
            v2f vert(appdata v)
            {
                v2f o;
                //把顶点信息转化到裁剪坐标下
                o.pos = UnityObjectToClipPos(v.vertex);
                o.uv = v.uv;
                return o;
            }
            fixed4 frag(v2f i) : SV_Target
            {
                fixed4 mainTex = tex2D(_MainTex,i.uv);
                return  mainTex * _Color;
            }
            
            ENDCG
        }
    }
}

测试脚本

复制代码
public class ShaderMaterialPropertyBlockTest : MonoBehaviour
    {
        [Header("生成的对象")] public GameObject gameObj;
        [Header("生成数量")] public int count = 100;
        [Header("生成范围")] public float range = 10;

        private GameObject [] gameObjects;
        private MaterialPropertyBlock prop;
        
        void Start()
        {
            gameObjects = new GameObject[count];
            prop = new MaterialPropertyBlock();

            for (int i = 0;i < count;i++) 
            {
                //随机位置并生成对象
                Vector2 pos = Random.insideUnitCircle * range;
                GameObject go = Instantiate(gameObj, new Vector3(pos.x, 0, pos.y),Quaternion.identity);
                gameObjects[i] = go;
            }
        }

        
        void Update()
        {
            //优化前,直接修改Shader中暴露的属性
            for (int i = 0;i < gameObjects.Length;i++)
            {
                float r = Random.Range(0f,1f);
                float g = Random.Range(0f,1f);
                float b = Random.Range(0f,1f);
                Color newColor = new Color(r, g, b, 1);
                gameObjects[i].GetComponentInChildren<MeshRenderer>().material.SetColor("_Color",newColor);
            }
            
            
        }
    }

效果(window->Analysis->Profilte):

二、使用[PerRendererData]标签,可以在脚本中使用SetPropertyBlock()对使用同一材质球的不同物体进行修改其Shader属性

测试脚本

复制代码
public class ShaderMaterialPropertyBlockTest : MonoBehaviour
    {
        [Header("生成的对象")] public GameObject gameObj;
        [Header("生成数量")] public int count = 100;
        [Header("生成范围")] public float range = 10;

        private GameObject [] gameObjects;
        private MaterialPropertyBlock prop;
        
        void Start()
        {
            gameObjects = new GameObject[count];
            prop = new MaterialPropertyBlock();

            for (int i = 0;i < count;i++) 
            {
                //随机位置并生成对象
                Vector2 pos = Random.insideUnitCircle * range;
                GameObject go = Instantiate(gameObj, new Vector3(pos.x, 0, pos.y),Quaternion.identity);
                gameObjects[i] = go;
            }
        }

        
        void Update()
        {
            //优化前,直接修改Shader中暴露的属性
            /*for (int i = 0;i < gameObjects.Length;i++)
            {
                float r = Random.Range(0f,1f);
                float g = Random.Range(0f,1f);
                float b = Random.Range(0f,1f);
                Color newColor = new Color(r, g, b, 1);
                gameObjects[i].GetComponentInChildren<MeshRenderer>().material.SetColor("_Color",newColor);
            }*/
            
            //优化后,使用MaterialPropertyBlock方案
            //该方案需要在Shader暴露属性中加上  [PerRendererData] 标签
            //这个是配合 SetPropertyBlock(prop);来使用的
            //这个可以用来更改同样材质球,都是不同物体的Shader属性
            for (int i = 0;i < gameObjects.Length;i++)
            {
                float r = Random.Range(0f,1f);
                float g = Random.Range(0f,1f);
                float b = Random.Range(0f,1f);
                Color newColor = new Color(r, g, b, 1);

                var mr = gameObjects[i].GetComponentInChildren<MeshRenderer>();
                mr.GetPropertyBlock(prop);
                prop.SetColor("_Color",newColor);
                mr.SetPropertyBlock(prop);
            }
        }
    }

优化后效果:

相关推荐
天人合一peng2 小时前
unity 生成标记根据背景色标记变色
unity·游戏引擎
天人合一peng6 小时前
unity 生成标记根据背景色变色为明显的颜色
unity·游戏引擎
魔士于安6 小时前
Unity 超市总动员 超市收银台 超市货架 超市购物手推车 超市常见商品
游戏·unity·游戏引擎·贴图·模型
CandyU26 小时前
Unity —— 数据持久化
unity·游戏引擎
zh路西法6 小时前
【Unity实现Oneshot胶卷显形】游戏窗口化与Win32API的使用
游戏·unity·游戏引擎
迪捷软件7 小时前
显控系统虚拟仿真的工程化路径
游戏引擎·cocos2d
凡情11 小时前
android隐私合规检测
android·unity
小贺儿开发11 小时前
Unity3D 本地 Stable Diffusion 文生图效果演示
人工智能·unity·stable diffusion·文生图·ai绘画·本地化
Swift社区12 小时前
传统游戏引擎 vs 鸿蒙 System 架构
架构·游戏引擎·harmonyos
mxwin1 天前
Unity Shader 半透明物体为什么不能写入深度缓冲?
unity·游戏引擎·shader