《Unity Shader入门精要》学习1:Phong 模型中法向量归一化的正确位置

学习《Unity Shader入门精要》的Phong模型时,按自己的理解写出了这样的代码:

复制代码
Shader "CH6/SpecularPixelLevel"
{
    Properties
    {
        _Diffuse("Diffuse", Color) = (1, 1, 1, 1)
        _Specular("Specular", Color) = (1, 1, 1, 1)
        _Gloss("Gloss", Range(8.0, 256)) = 20
    }
    SubShader
    {
        Pass
        {
            Tags { "RenderType"="ForwardBase" }
    
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "Lighting.cginc"

            fixed4 _Diffuse;
            fixed4 _Specular;
            float _Gloss;
            struct appdata_t
            {
                float4 pos: POSITION;
                float3 normal: NORMAL;
            };
            
            struct v2f
            {
                float4 pos: SV_POSITION;
                fixed4 worldPos: TEXCOORD0;
                float3 worldNormal: TEXCOORD1;
            };
    
            v2f vert(appdata_t v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.pos);
                o.worldPos = mul(unity_ObjectToWorld, v.pos);
                o.worldNormal = normalize(mul(v.normal, (fixed3x3)unity_WorldToObject));

                return o;
            }

            float4 frag(v2f i): SV_Target
            {
				// ambient
                fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz;
                
                // diffuse
                fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz);
                fixed3 diffuse = _Diffuse.rgb * _LightColor0.rgb * saturate(dot(i.worldNormal, worldLightDir));

                // specular
                fixed3 reflectDif = reflect(-worldLightDir, i.worldNormal);
                fixed3 viewDir = normalize(_WorldSpaceCameraPos - i.worldPos);
                fixed3 specular = _Specular.rgb * _LightColor0.rgb * pow(saturate(dot(reflectDif, viewDir)), _Gloss);

                fixed3 color = ambient + diffuse + specular;
                
                return fixed4(color, 1);
            }
            ENDCG
        }
    }
    FallBack "Diffuse"
}

结果代码和书上的不一样,书上在vertex shader没有将worldNormal归一化,归一化放在了fragment shader。我的代码是将顶点数据的worldNormal归一化了,所以片元数据插值后也是归一化的嘛?于是就想验证一下自己的想法,比较代码也挺简单:

复制代码
Shader "CH6/SpecularPixelLevel"
{
    Properties
    {
        _Diffuse("Diffuse", Color) = (1, 1, 1, 1)
        _Specular("Specular", Color) = (1, 1, 1, 1)
        _Gloss("Gloss", Range(8.0, 256)) = 20
    }
    SubShader
    {
        Pass
        {
            Tags { "RenderType"="ForwardBase" }
    
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "Lighting.cginc"

            fixed4 _Diffuse;
            fixed4 _Specular;
            float _Gloss;
            struct appdata_t
            {
                float4 pos: POSITION;
                float3 normal: NORMAL;
            };
            
            struct v2f
            {
                float4 pos: SV_POSITION;
                fixed4 worldPos: TEXCOORD0;
                float3 worldNormal: TEXCOORD1;
            };
    
            v2f vert(appdata_t v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.pos);
                o.worldPos = mul(unity_ObjectToWorld, v.pos);
                o.worldNormal = normalize(mul(v.normal, (fixed3x3)unity_WorldToObject));

                return o;
            }

            float4 frag(v2f i): SV_Target
            {

				float worldNormalLength = length(i.worldNormal);
                float diff = abs(worldNormalLength - 1.0);
                fixed3 color = lerp(fixed3(1, 0, 0), fixed3(0, 1, 0), step(0.001, diff));
                return fixed4(color, 1);
            }
            ENDCG
        }
    }
    FallBack "Diffuse"
}

代码通过length(i.worldNormal)计算法向量长度

,abs(normalLength - 1.0)获取与 1 的偏差,

当偏差小于 0.001(近似为 1)时显示绿色,否则显示红色。结果显示

调整scale比例,长度不是1的法向量更多,但计算光照还是需要长度是1的法向量的:

书上的代码是没有问题的,vertex shader顶点数据不用归一化直接插值,插值后的片元数据再进行归一化。附上正确的代码:

复制代码
Shader "CH6/SpecularPixelLevel"
{
    Properties
    {
        _Diffuse("Diffuse", Color) = (1, 1, 1, 1)
        _Specular("Specular", Color) = (1, 1, 1, 1)
        _Gloss("Gloss", Range(8.0, 256)) = 20
    }
    SubShader
    {
        Pass
        {
            Tags { "RenderType"="ForwardBase" }
    
            CGPROGRAM
            #pragma vertex vert
            #pragma fragment frag
            #include "Lighting.cginc"

            fixed4 _Diffuse;
            fixed4 _Specular;
            float _Gloss;
            struct appdata_t
            {
                float4 pos: POSITION;
                float3 normal: NORMAL;
            };
            
            struct v2f
            {
                float4 pos: SV_POSITION;
                fixed4 worldPos: TEXCOORD0;
                float3 worldNormal: TEXCOORD1;
            };
    
            v2f vert(appdata_t v)
            {
                v2f o;
                o.pos = UnityObjectToClipPos(v.pos);
                o.worldPos = mul(unity_ObjectToWorld, v.pos);
                o.worldNormal = mul(v.normal, (fixed3x3)unity_WorldToObject);

                return o;
            }

            float4 frag(v2f i): SV_Target
            {

                // ambient
                fixed3 worldNormal = normalize(i.worldNormal);
                fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.xyz;
                
                // diffuse
                fixed3 worldLightDir = normalize(_WorldSpaceLightPos0.xyz);
                fixed3 diffuse = _Diffuse.rgb * _LightColor0.rgb * saturate(dot(worldNormal, worldLightDir));

                // specular
                fixed3 reflectDif = reflect(-worldLightDir, worldNormal);
                fixed3 viewDir = normalize(_WorldSpaceCameraPos - i.worldPos);
                fixed3 specular = _Specular.rgb * _LightColor0.rgb * pow(saturate(dot(reflectDif, viewDir)), _Gloss);

                fixed3 color = ambient + diffuse + specular;
                
                return fixed4(color, 1);
            }
            ENDCG
        }
    }
    FallBack "Diffuse"
}
相关推荐
大龄程序员狗哥8 小时前
第25篇:Q-Learning算法解析——强化学习中的经典“价值”学习(原理解析)
人工智能·学习·算法
南境十里·墨染春水9 小时前
linux学习进展 线程同步——互斥锁
java·linux·学习
nashane10 小时前
HarmonyOS 6学习:旋转动画优化与长截图性能调优——打造丝滑交互体验的深度实践
学习·交互·harmonyos·harmonyos 5
华清远见IT开放实验室10 小时前
智能手表完整项目实现,比赛求职双向加分,基于嵌入式大赛推荐开发板(STM32U5)
stm32·单片机·嵌入式硬件·学习·智能手表·嵌入式大赛
炽烈小老头10 小时前
【 每天学习一点算法 2026/04/22】四数相加 II
学习·算法
uncle_ll11 小时前
LangChain基础学习笔记
笔记·学习·langchain·llm·rag
三品吉他手会点灯11 小时前
C语言学习笔记 - 14.C编程预备计算机专业知识 - 本讲内容概述
c语言·笔记·学习
Thanwind11 小时前
从0开始的机器学习之旅(二):监督学习,从线性回归说起
学习·机器学习·线性回归
2501_9423264411 小时前
易速乐考,轻松备考
学习·教育电商
菜鸟‍11 小时前
【CVPR 2026】LitePT:更轻、更强的点云 Transformer【论文学习】
深度学习·学习·transformer