Unity3D Shader 顶点和片段着色器

Unity3D Shader 顶点和片段着色器

Unity 的自定义 Shader 一般由两部分最核心的程序组成:

  • 顶点着色器(Vertex Shader,简称 vert)
  • 片段着色器(Fragment Shader,简称 frag)

如果你刚入门 Shader,这两个函数是你最先要理解的东西,因为所有渲染行为都从这里开始。

什么是顶点着色器(vert)

顶点着色器(vertex shader)负责"处理网格的每一个顶点"。

Unity Mesh 的每个点(MeshFilter 里的那个 Mesh)都会被送入顶点着色器,Shader 内的 appdata 就是顶点的结构体。

它主要负责:

将对象空间坐标转换到裁剪空间

也就是:

  • 本地坐标 → 世界坐标
  • 世界坐标 → 视图空间(Camera)
  • 视图空间 → 裁剪空间(Clip Space)

最终输出给 GPU 进行光栅化。

在 Unity 中通常使用:

c 复制代码
UnityObjectToClipPos(v.vertex)

传递数据给片段着色器(颜色 / UV / 法线等)

顶点着色器可以将数据装进 v2f(vertex-to-fragment)结构体中,传递到片段着色器。

例如传递 UV:

c 复制代码
o.uv = v.uv;

可以修改顶点位置(顶点动画)

例如:

  • 顶点扭曲
  • 呼吸效果
  • 旗帜随风摆动
  • 水波

顶点动画就是在这里实现的。

什么是片段着色器(frag)

片段着色器(fragment shader)负责"处理每个像素最后显示什么颜色"。

它的输入是经过光栅化之后的"片段"(像素点),每个 Pixel 都会跑一次 frag()

片段着色器通常负责:

计算最终颜色

包含:

  • 采样贴图(Texture)
  • 颜色混合
  • Light / BRDF
  • 各类特效(溶解、描边、灰度化、等)

例如返回一个固定颜色:

c 复制代码
return fixed4(1, 0, 0, 1); // 红色

采样纹理

c 复制代码
fixed4 col = tex2D(_MainTex, i.uv);

片段着色器比顶点着色器更耗性能

因为屏幕可能有 百万级像素 ,每个像素都要跑 frag,所以:

  • 顶点越少 = 越省性能
  • 片段越复杂 = 越吃性能

如果你做的是 UI、全屏特效、后处理,更要注意 fragment 的复杂度。

最基础的 vert / frag Shader

下面代码是 Unity Shader 中最基础的顶点 + 片段结构。只做两件事:

  • vert:把顶点从模型空间变成裁剪空间
  • frag:返回 _Color 颜色作为像素颜色
c 复制代码
Shader "Custom/VertFragDemo"
{
    Properties
    {
        _Color("Main Color", Color) = (1,1,1,1)
    }

    SubShader
    {
        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag

            #include "UnityCG.cginc"

            struct appdata
            {
                float4 vertex : POSITION;
            };

            struct v2f
            {
                float4 pos : SV_POSITION;
            };

            fixed4 _Color;

            v2f vert (appdata v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.vertex);
                return o;
            }

            fixed4 frag (v2f i) : SV_Target
            {
                return _Color;
            }
            ENDCG
        }
    }
}

可视化理解:顶点着色 → 光栅化 → 片段着色

以下是 GPU 渲染的基本流程,非常重要:

c 复制代码
 Mesh 顶点数据
       ↓
 顶点着色器(vert)
       ↓
 裁剪、组装三角形
       ↓
 光栅化(生成像素)
       ↓
 片段着色器(frag)
       ↓
 Framebuffer(最终图像)

你可以看到:

  • vert 运行次数:顶点数量(通常比较少)
  • frag 运行次数:像素数量(可能是百万级)

所以片段着色器比顶点着色器更重要、也更昂贵。

常用的 appdata / v2f 字段

appdata(从 Mesh 输入)

c 复制代码
struct appdata
{
    float4 vertex : POSITION;
    float2 uv     : TEXCOORD0;
    float3 normal : NORMAL;
};

v2f(传递给 frag)

c 复制代码
struct v2f
{
    float4 pos : SV_POSITION;
    float2 uv  : TEXCOORD0;
};

你想传什么给 frag 就在这里声明。

实战:加一点变化(UV 贴图 + 色调)

下面是一个同时使用顶点 + 片段 + 贴图采样的版本:

c 复制代码
Shader "Custom/TextureTint"
{
    Properties
    {
        _MainTex("Texture", 2D) = "white" {}
        _Tint("Tint Color", Color) = (1,1,1,1)
    }

    SubShader
    {
        Pass
        {
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag

            #include "UnityCG.cginc"

            struct appdata
            {
                float4 vertex : POSITION;
                float2 uv : TEXCOORD0;
            };

            struct v2f
            {
                float4 pos : SV_POSITION;
                float2 uv : TEXCOORD0;
            };

            sampler2D _MainTex;
            float4 _MainTex_ST;
            fixed4 _Tint;

            v2f vert(appdata v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.vertex);
                o.uv = TRANSFORM_TEX(v.uv, _MainTex);
                return o;
            }

            fixed4 frag(v2f i) : SV_Target
            {
                fixed4 col = tex2D(_MainTex, i.uv);
                return col * _Tint;
            }
            ENDCG
        }
    }
}

你可以:

  • 给模型贴一张纹理
  • _Tint 调节颜色

这是一个最常用的 vert/frag 模板,几乎所有 Unity 自定义 Shader 都从这里开始。

总结

内容 顶点着色器(vert) 片段着色器(frag)
输入 Mesh 顶点数据 每个像素
功能 坐标变换、传递数据、顶点动画 计算最终颜色、采样纹理
运行次数 顶点数量 像素数量(大量)
性能影响
典型用途 旗帜飘动、水波、形变 灯光、贴图、特效、颜色计算
相关推荐
weixin_424294675 天前
Unity 调用Steamworks API 的 SteamUserStats.RequestCurrentStats()报错
unity·游戏引擎·steamwork
HoFunGames5 天前
Unity小地图,Easy Minimap System MT-GPS插件
unity·游戏引擎
wy3258643645 天前
Unity 新输入系统InputSystem(基本操作)
unity·c#·游戏引擎
WarPigs5 天前
着色器multi_compile笔记
unity·着色器
ECHO飞跃 0125 天前
Unity2019 本地推理 通义千问0.5-1.5B微调导入
人工智能·深度学习·unity·llama
Unity游戏资源学习屋5 天前
【Unity UI资源包】GUI Pro - Casual Game 专为休闲手游打造的专业级UI资源包
ui·unity
冰凌糕5 天前
Unity3D Shader 顶点法线外扩实现描边效果
unity
小菱形_5 天前
【Unity】TimeLine
unity·游戏引擎
小贺儿开发6 天前
Unity3D 自动化物流分拣模拟
运维·科技·unity·自动化·人机交互·传送带·物流分拣
EQ-雪梨蛋花汤6 天前
【3D可视化】基于 Unity 的智慧体育馆三维信息可视化大屏实践
3d·unity·信息可视化