简单了解一下Unity的MaterialPropertyBlock

MaterialPropertyBlock是Unity提供的一个类,用于在运行时动态修改渲染器的材质属性,而无需为每个渲染器创建新的材质实例。它通过为每个渲染器设置独立的属性值,实现在共享材质基础上的个性化定制。这种方式既能满足灵活性需求,又能减少内存开销和渲染批次,是优化性能的重要工具。

MaterialPropertyBlock是一个轻量级容器,用于存储材质属性的覆盖值,而无需创建新的材质实例。它允许你:

  1. 为单个Renderer实例设置特定的材质属性值
  2. 保持原始材质的共享引用
  3. 维护GPU批处理的优势

在内部,Unity使用这些属性块作为渲染命令的一部分,在绘制特定对象时临时覆盖共享材质的属性。

MaterialPropertyBlock在以下场景中尤为实用:

  1. 批量物体的个性化定制
    当场景中有大量相似物体(例如草地、树木或建筑),它们共享同一个材质,但需要略微不同的属性(如颜色、纹理偏移)时,MaterialPropertyBlock可以在不增加材质实例的情况下,为每个物体设置独特的属性。
  2. 动态效果
    在运行时需要频繁调整材质属性的情况,例如:
    • 角色的高光效果随时间变化。
    • UI元素的颜色渐变动画。
    • 特效的透明度或亮度调整。
  3. 优化渲染性能
    通过减少材质实例的数量,MaterialPropertyBlock可以降低Draw Call(渲染调用)次数,尤其在移动平台等性能敏感的环境中效果显著。

MaterialPropertyBlock与material属性的区别

在Unity中,渲染器的material属性和MaterialPropertyBlock都可以修改材质属性,但它们的工作方式和适用场景有很大不同。

material属性

  • 行为:当你访问或修改renderer.material时,Unity会为该渲染器创建一个新的材质实例(如果之前没有独立实例)。
  • 影响:对这个材质实例的修改只影响当前渲染器,但会增加内存开销和渲染批次。
  • 适用场景:适合少量物体需要完全独特的材质属性时使用。

MaterialPropertyBlock

  • 行为:通过SetPropertyBlock方法,MaterialPropertyBlock为渲染器设置独立的属性值,但渲染器仍然共享底层的材质资源,不会创建新实例。
  • 影响:属性修改仅影响当前渲染器,且不会增加内存或批次开销。
  • 适用场景:适合大量物体需要个性化属性但希望保持材质共享的情况。

区别总结

特性 material属性 MaterialPropertyBlock
材质实例 创建新的材质实例 不创建新实例,共享材质
内存开销 较高(每个实例占用内存) 较低(共享材质)
渲染批次 可能增加(不同材质实例) 较少(共享材质,属性差异不影响批次)
适用场景 少量物体的独特属性 大量物体的个性化属性

使用示例

基本使用

以下是一个简单的示例,展示如何使用MaterialPropertyBlock为物体设置独特的颜色:

cs 复制代码
using UnityEngine;

public class ColorChanger : MonoBehaviour
{
    private Renderer renderer;
    private MaterialPropertyBlock propBlock;

    void Start()
    {
        renderer = GetComponent<Renderer>();
        propBlock = new MaterialPropertyBlock();
        // 设置颜色
        propBlock.SetColor("_Color", Color.red);
        renderer.SetPropertyBlock(propBlock);
    }
}

在这个例子中,物体的颜色被设置为红色,但不会创建新的材质实例。

动态修改属性

你可以在运行时动态更新MaterialPropertyBlock中的属性值,例如实现颜色渐变:

cs 复制代码
void Update()
{
    // 每帧随机改变颜色
    Color newColor = new Color(Random.value, Random.value, Random.value);
    propBlock.SetColor("_Color", newColor);
    renderer.SetPropertyBlock(propBlock);
}

支持的属性类型

MaterialPropertyBlock支持多种属性设置方法,包括:

  • SetColor:设置颜色。
  • SetFloat:设置浮点数。
  • SetVector:设置向量。
  • SetTexture:设置纹理。
  • SetMatrix:设置矩阵。
  • SetBuffer: 设置缓冲区。

这些方法可以满足大多数材质属性的修改需求。

cs 复制代码
// 各种属性类型设置示例
void SetVariousProperties()
{
    // 常见标量和向量
    propertyBlock.SetFloat("_Metallic", 0.8f);
    propertyBlock.SetVector("_SpecColor", new Vector4(0.9f, 0.8f, 0.1f, 1.0f));
    propertyBlock.SetColor("_EmissionColor", Color.red * 2.0f);
    
    // 纹理属性
    propertyBlock.SetTexture("_MainTex", customTexture);
    propertyBlock.SetTexture("_BumpMap", customNormalMap);
    
    // 矩阵属性
    Matrix4x4 texTransform = Matrix4x4.TRS(
        new Vector3(0.5f, 0.5f, 0), 
        Quaternion.Euler(0, 0, 45), 
        Vector3.one
    );
    propertyBlock.SetMatrix("_TextureMatrix", texTransform);
    
    // 缓冲区属性 (Unity 2019.3+)
    propertyBlock.SetBuffer("_InstanceData", computeBuffer);
    
    // 使用属性ID(性能更优)
    int emissionColorID = Shader.PropertyToID("_EmissionColor");
    propertyBlock.SetColor(emissionColorID, Color.blue * 1.5f);
}

性能优化建议

  1. 批量处理
    对于大量物体,使用MaterialPropertyBlock可以避免创建多个材质实例,从而减少Draw Call。确保所有物体共享同一个材质,仅通过MaterialPropertyBlock调整差异化属性。
  2. 重复使用实例
    MaterialPropertyBlock对象可以重复使用,避免在循环或频繁调用中创建新实例,以减少性能开销。
  3. 结合sharedMaterial
    如果需要对所有物体进行全局属性调整,可以使用renderer.sharedMaterial修改共享材质,而将个性化设置交给MaterialPropertyBlock。
  4. 检查属性名称
    设置属性时,需确保使用Shader中定义的正确属性名称(如_Color),否则修改将无效。
相关推荐
死也不注释11 小时前
【鸡零狗碎记录】
unity·c#
★YUI★14 小时前
学习游戏制作记录(剑投掷技能)7.26
学习·游戏·unity·c#
★YUI★1 天前
学习游戏制作记录(克隆技能)7.25
学习·游戏·unity·c#
不绝1912 天前
ARPG开发流程第一章——方法合集
算法·游戏·unity·游戏引擎
玩代码2 天前
Unity里的加力
开发语言·unity
贵州晓智信息科技2 天前
Unity 性能优化全攻略
unity·性能优化·游戏引擎
UWA2 天前
UWA DAY 2025 游戏开发者大会|全议程
游戏·unity·性能优化·游戏开发·uwa·unreal engine
unicrom_深圳市由你创科技2 天前
Unity 的UI动画调节
ui·unity·游戏引擎
咩咩觉主2 天前
Unity编辑器拓展 IMGUI与部分Utility知识总结(代码+思维导图)
unity·c#·编辑器·游戏引擎
龚子亦2 天前
【Unity开发】数据存储——XML
xml·unity·游戏引擎·数据存储·游戏开发