news 2026/4/18 7:44:50

【URP】Unity[视差遮挡贴图]原理剖析实践

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
【URP】Unity[视差遮挡贴图]原理剖析实践

视差遮挡贴图(Parallax Occlusion Mapping, POM)介绍

视差遮挡贴图是视差贴图技术的高阶实现,通过‌光线步进(Raymarching)算法‌精确计算视线与高度图的交点,模拟复杂表面(如砖墙、岩石)的几何遮挡效果。相比标准视差贴图,POM能更真实地表现深度变化和自阴影,适用于高精度材质表现。

核心原理

‌分层深度检测‌将视线方向在切线空间分解为多层(通常8-15层),逐层采样高度图,通过二分法逼*视线与表面的交点。

‌动态采样优化‌根据视角与法线的夹角动态调整采样层数(**行视角增加层数,垂直视角减少层数),*衡精度与性能。

‌遮挡关系重建‌通过交点处的UV偏移修正纹理采样位置,模拟凹凸表面的光线遮挡效果。

Unity URP 实现示例与原理详解

代码关键点解析

‌动态采样层数‌

通过lerp(_MaxSamples, _MinSamples, saturate(dot(float3(0,0,1), viewDirTS)))实现视角自适应分层,优化性能。

‌光线步进循环‌

循环比较当前层高度与采样深度,找到首个交点区间,避免全精度遍历。

‌二分法优化‌

在初步交点区间内使用lerp插值计算精确UV,减少采样次数。

ParallaxOcclusionMapping.shader

Shader "Universal Render Pipeline/POM"

{

Properties

{

_MainTex("Albedo", 2D) = "white" {}

_NormalMap("Normal Map", 2D) = "bump" {}

_HeightMap("Height Map", 2D) = "white" {}

_ParallaxScale("Height Scale", Range(0, 0.1)) = 0.05

_MinSamples("Min Samples", Int) = 8

_MaxSamples("Max Samples", Int) = 15

}

SubShader

{

Tags { "RenderPipeline"="UniversalPipeline" }

HLSLINCLUDE

#include "Packages/com.unity.render-pipelines.universal/ShaderLibrary/Core.hlsl"

TEXTURE2D(_MainTex); SAMPLER(sampler_MainTex);

TEXTURE2D(_NormalMap); SAMPLER(sampler_NormalMap);

TEXTURE2D(_HeightMap); SAMPLER(sampler_HeightMap);

float _ParallaxScale;

int _MinSamples, _MaxSamples;

float2 ParallaxOcclusionMapping(float3 viewDirTS, float2 uv)

{

// 动态计算采样层数

int numSamples = (int)lerp(_MaxSamples, _MinSamples, saturate(dot(float3(0,0,1), viewDirTS)));

float layerHeight = 1.0 / numSamples;

float2 deltaUV = _ParallaxScale * viewDirTS.xy / viewDirTS.z / numSamples;

// 光线步进初始化

float currentLayerHeight = 0;

float2 currentUV = uv;

float currentDepth = 1 - SAMPLE_TEXTURE2D(_HeightMap, sampler_HeightMap, currentUV).r;

// 分层检测

[loop]

for (int i = 0; i < 15; ++i) {

if (currentLayerHeight >= currentDepth) break;

currentUV -= deltaUV;

currentDepth = 1 - SAMPLE_TEXTURE2D(_HeightMap, sampler_HeightMap, currentUV).r;

currentLayerHeight += layerHeight;

}

// 二分法精确交点

float2 prevUV = currentUV + deltaUV;

float prevDepth = currentDepth - layerHeight;

float weight = (currentLayerHeight - currentDepth) / (prevDepth - currentDepth);

return lerp(currentUV, prevUV, weight);

}

ENDHLSL

Pass

{

HLSLPROGRAM

#pragma vertex vert

#pragma fragment frag

struct Attributes

{

float4 positionOS : POSITION;

float2 uv : TEXCOORD0;

float3 normalOS : NORMAL;

float4 tangentOS : TANGENT;

};

struct Varyings

{

float4 positionCS : SV_POSITION;

float2 uv : TEXCOORD0;

float3 viewDirTS : TEXCOORD1;

};

Varyings vert(Attributes IN)

{

Varyings OUT;

VertexPositionInputs posInput = GetVertexPositionInputs(IN.positionOS.xyz);

OUT.positionCS = posInput.positionCS;

// 转换视角方向到切线空间

VertexNormalInputs normInput = GetVertexNormalInputs(IN.normalOS, IN.tangentOS);

float3 viewDirWS = GetWorldSpaceViewDir(posInput.positionWS);

OUT.viewDirTS = TransformWorldToTangent(viewDirWS,

normInput.tangentWS, normInput.bitangentWS, normInput.normalWS);

OUT.uv = IN.uv;

return OUT;

}

half4 frag(Varyings IN) : SV_Target

{

// 计算POM偏移后的UV

float2 pomUV = ParallaxOcclusionMapping(normalize(IN.viewDirTS), IN.uv);

// 采样最终纹理

half4 albedo = SAMPLE_TEXTURE2D(_MainTex, sampler_MainTex, pomUV);

half3 normalTS = UnpackNormal(SAMPLE_TEXTURE2D(_NormalMap, sampler_NormalMap, pomUV));

return half4(albedo.rgb, 1);

}

ENDHLSL

}

}

}

技术对比与性能建议

特性 标准视差贴图 陡峭视差贴图 POM

‌采样次数‌ 单次 5-15次 8-15次+二分法

‌陡峭表面精度‌ 低 中 高

‌适用*台‌ 移动端 PC/主机 高端PC

‌推荐参数‌ Scale=0.02 Scale=0.05 Scale=0.03-0.07

实际应用中需注意:

高度图建议使用法线贴图的Alpha通道节省资源

过高的_ParallaxScale可能导致边缘拉伸,建议不超过0.1

移动端可减少_MaxSamples至8层以降低开销

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/14 6:17:25

0x3f第十天复习(考研日2)(9.18-12.30,14.00-15.00)

二叉搜索树验证 前序2min ac4min ac4min ac1min ac二叉搜索树验证 中序 6min x 基本没问题&#xff0c;记得 每次递归都要return 结果 6min ac 4min ac3min ac二叉搜索树验证 后序 30min x 最后return min(lmin,x), max(rmax,x) 还是有点没理解 15min ac 10min x还是不理解 (r…

作者头像 李华
网站建设 2026/4/9 8:14:01

医疗AI智能体架构设计:六大核心模块与七种专业智能体类型全解析

文章介绍了医疗AI智能体的六大核心模块框架&#xff1a;感知、对话接口、交互系统、工具集成、记忆学习和推理&#xff0c;以及七种专业智能体类型的特点与应用场景。这一模块化架构旨在构建安全、可解释且自适应的医疗AI系统&#xff0c;推动人工智能在医疗领域的深度应用&…

作者头像 李华
网站建设 2026/4/17 6:38:33

js函数声明和函数表达式的理解

在JS中,函数声明会被提升&#xff0c;这意味着函数可以在声明之前被调用。当你使用函数声明的方式定义函数: function resizeFn() {...}整个函数声明会被提升到作用域的顶部。这意味着在整个作用域内&#xff0c;无论函数在何处声明&#xff0c;都可以在声明前调用。函数声明会…

作者头像 李华
网站建设 2026/4/18 3:27:40

奶奶都能看懂的 C++ —— vector 与迭代器

但是在讲解它之前&#xff0c;我们需要先了解迭代的对象是什么。常见的一种&#xff0c;叫做 vector。vector 类型使用可变有序序列我们知道&#xff0c;数学里&#xff0c;vector 是向量的意思。但 C 里的向量和它不太一样。它的含义是&#xff0c;具有可变元素个数的有序对象…

作者头像 李华
网站建设 2026/4/18 3:37:33

基于java的SpringBoot/SSM+Vue+uniapp的校园活动管理系统的详细设计和实现(源码+lw+部署文档+讲解等)

文章目录前言详细视频演示具体实现截图技术栈后端框架SpringBoot前端框架Vue持久层框架MyBaitsPlus系统测试系统测试目的系统功能测试系统测试结论为什么选择我代码参考数据库参考源码获取前言 &#x1f31e;博主介绍&#xff1a;✌全网粉丝15W,CSDN特邀作者、211毕业、高级全…

作者头像 李华