Unity Shader 法线贴图的七种错误用法

在 Unity URP 管线中,法线贴图是最常被误用的技术之一。
本文系统梳理从导入设置到 Shader 代码的全链路高频错误。

01贴图导入时未勾选 Normal Map 类型

这是最基础也最容易被忽视的错误。Unity 需要知道贴图是法线贴图,才能在导入时执行正确的解压和色彩空间处理。如果 Texture Type 仍设置为默认的 Default,Unity 会将其当作普通的 sRGB 颜色贴图处理,导致法线信息完全错误。

Inspector 中的正确 vs 错误设置对比

症状

物体表面完全看不到法线凹凸效果,或光照方向异常(比如背光面反而很亮)。Unity Inspector 也会显示黄色警告:"Texture type is not set to Normal map"

修复

在 Inspector → Texture Type → Normal Map,点 Apply。移动端建议同时开启 Crunch Compression 配合 ETC2 格式降低内存。

02在 sRGB 色彩空间中采样法线

法线贴图存储的是方向向量(线性数据),不是颜色。如果项目开启了 Linear Color Space(URP 标准要求),而采样时走了 sRGB 的 Gamma 矫正路径,向量值就会被幂函数扭曲,导致法线方向偏移。

sRGB 矫正对法线向量的扭曲示意

在 URP Shader 中,属性块要声明为 [NoScaleOffset],采样器类型应使用 TEXTURE2D 而非带 Gamma 自动矫正的普通 sampler2D。更关键的是 UnpackNormal 本身假设贴图已经是线性数据,如果上游走了 sRGB 解码,结果就错了。

注意

导入设置中将 Texture Type 设为 Normal Map 后,Unity 会自动禁用 sRGB 标志位。但如果在代码里用了 sampler2D 并搭配旧版宏,依然可能在 Built-in 遗留路径中触发 gamma 矫正。


03忘记将法线从切线空间转换到世界空间

法线贴图默认存储在 切线空间(Tangent Space) 中。蓝色偏紫的贴图外观正是因为法线大体朝向 Z 轴(0.5, 0.5, 1.0)。如果直接把从贴图中解出的向量当作世界空间法线使用,光照计算就会完全错乱——物体旋转后法线不跟着走。

三种法线空间的关系

URP 快捷方式

使用 GetWorldSpaceNormalizeViewDir 和 TransformTangentToWorld(来自 SpaceTransforms.hlsl),URP 已内置完整的空间转换函数族,不必手动构建矩阵。


04未对法线向量归一化

从顶点着色器插值到片段着色器的法线,以及经过矩阵变换后的法线,插值和矩阵乘法都会破坏向量的单位长度。如果不重新归一化,光照公式中的 dot product 计算就会产生错误的亮度值——尤其在低精度 half 类型下非常明显。

插值导致法线长度偏离单位球示意

05BC5 / DXT5nm 压缩格式重建 Z 分量错误

移动端常用 ETC2,PC/主机常用 BC5(ATI2N)来压缩法线贴图。这两种格式只存储 XY 两个通道,Z 分量需要在 Shader 中用勾股定理重建:

float z = sqrt(1.0 - saturate(dot(xy, xy)));

常见错误是使用了错误的通道映射,或忘记 saturate 导致负数开方产生 NaN。

DXT5nm 特殊情况

DXT5nm(旧版 Unity Built-in 默认格式)将 X 存在 Alpha 通道,Y 存在 Green 通道。UnpackNormalmapRGorAG 宏会自动处理,但手写 Shader 时必须注意区分 .ag 而非 .rg。URP 新项目建议直接使用 BC5 并调用 UnpackNormal,不要手动拆通道。


06缩放法线强度后未重新归一化

材质里的 Normal Map Scale(法线强度)在 Shader 中通常是乘以 XY 分量来实现:强度越大,表面越"凹凸"。但只缩放 XY 而 Z 不变,向量就不再是单位长度,必须重新归一化。

Unity URP 内置的 UnpackNormalScale(sample, scale) 函数已经正确处理了这个问题,推荐直接使用:

// URP 推荐方式:一步到位
float3 normalTS = UnpackNormalScale(normalSample, _NormalScale);

07在 URP 中混用 Built-in 的 UnpackNormal 宏

这是 Built-in 管线迁移到 URP 时最高发的错误之一。Built-in 的 UnpackNormal 在 UnityCG.cginc 中,而 URP 的版本在 Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl 中,两者对通道顺序和缩放处理并不完全一致

混用的典型症状

编译通过但运行时法线方向反向、凸起变凹陷,或在不同平台(PC vs Android)表现不一致。这往往在迁移旧项目时悄悄发生,因为两套 API 的函数名完全相同。

快速检查方法

在 Shader 里全局搜索 UnityCG.cgincLighting.cgincsampler2Dtex2D,凡是出现这些关键词的 URP Shader 都需要逐一审查替换。


总结错误速查表

#错误严重度症状修复要点
01导入类型未设为 Normal Map致命光照完全错乱,Inspector 警告Texture Type → Normal Map
02sRGB 色彩空间采样法线严重法线角度偏移,高光位置错误TEXTURE2D + 线性采样
03未转换到世界空间致命旋转物体后法线不随之变化TBN 矩阵 / TransformTangentToWorld
04法线未归一化中等亮度溢出、移动端高精度丢失每次使用前 normalize()
05压缩格式 Z 分量重建错误严重法线 Z 轴错误,可能出现 NaNsaturate + sqrt 重建,注意通道
06缩放后未重归一化中等法线强度调整后光照异常用 UnpackNormalScale()
07混用 Built-in / URP API严重多平台表现不一致,法线方向反转全换 URP Core.hlsl 头文件
软件概述 UG(Unigraphics NX)是一款由西门子(Siemens PLM Software)开发的交互式CAD/CAM/CAE系统。作为全球领先的产品工程解决方案,它集成了产品设计、工程仿真与制造加工于一体。其功能强大且应用广泛,能够轻松实现各种复杂实体和造型的构造,为模具、汽车、航空航天及通用机械等行业提供了高性能的机械设计与制图灵活性。 软件基础信息 • 支持系统: 64位 Windows 10、Windows 11 核心功能模块 一、创新设计:高效、灵活、无缝协同 全链路产品设计 涵盖从2D布局、3D建模、装配设计到图纸文档记录的各个环节,大幅提升设计吞吐量,缩短交付周期超35%。 强大的同步建模技术 打破数据壁垒,可无缝导入并直接修改来自其他CAD系统的几何模型,是跨平台协同设计的理想选择。 复杂装配管理 专为大型复杂产品打造,即使面对成千上万的零件也能从容应对,快速识别并解决数字样机中的干涉等问题。 集成设计验证 内置自动验证功能,实时监控设计是否符合公司及行业标准;结合PLM数据可视化合成,辅助工程师做出更明智的决策。 二、综合仿真(Simcenter 3D):精准预测,降低试错成本 极速前后处理 依托先进的几何引擎,将强大的分析命令与几何编辑紧密集成,相比传统有限元工具,可缩短高达70%的仿真建模时间。 全方位结构分析 在同一环境中集成线性静力学、动态、疲劳及非线性分析,底层由业界顶尖的NX Nastran解算器提供支持,确保计算的高精度与可靠性。 声学与热管理分析 提供内外声学仿真以优化音质、降低噪音;具备一流的热传导仿真能力,帮助电子产品和工业机械实现最佳热管理方案。 多物理场耦合 简化了结构动力学、热传导、流体流动等复杂物理现象的模拟过程,消除外部数据传输错误,真实还原产品运行工况。 三、智能制造(CAM):打通从计划到车间的数字主线 全面的制造解决方案 提供从工装设计、CAM编程到机床控制器(如Sinumerik)的一体化支持,助力制定更科学的生产决策。 深度集成的PLM环境 借助Teamcenter实现数据和流程的统一管理,避免多数据库冲突,支持重用验证过的加工工艺与刀具库。 车间级互联 通过DNC系统与车间无缝对接,直接将加工数据和刀具清单下发至CNC机床,实现计划与生产的紧密结合。 提质增效 优化NC编程与刀具路径,提升表面精加工水平与零件精度;减少人为错误,显著提高新机床部署成功率及制造资源利用率。 总结 UG NX 2023作为一款集成化的产品工程解决方案,通过其强大的设计、仿真和制造功能,为现代制造业提供了完整的数字化产品开发平台。无论是复杂产品的设计验证,还是精密制造的流程优化,UG NX 2023都能为工程师团队提供高效、可靠的解决方案,助力企业提升产品创新能力和市场竞争力。 适用领域 模具设计、汽车制造、航空航天、通用机械、消费电子等
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值