Lambert (兰伯特)光照模型
创始人
2024-03-01 01:23:32
0

漫反射的定义

漫反射是投射在粗糙表面上的光向各个方向反射的现象。当一束平行的入射光线射到粗糙的表面时,表面会把光线向着四面八方反射,所以入射线虽然互相平行,由于各点的法线方向不一致,造成反射光线向不同的方向无规则地反射,这种反射称之为“漫反射”或“漫射”。这种反射的光称为漫射光。

Lambert定律

漫反射光的强度近似地服从于Lambert定律,即漫反射光的光强与表面法线和光源方向之间的夹角的余弦成正比。
在这里插入图片描述

原理公式:diffuse = I*cosθ;

diffuse:反射光线的的光强;

I:入射光线的光强,方向如上图所示;

cosθ:光源方向和该顶点法线的余弦,法线方向 · 光源方向,cosθ = dot(L,N);

在Unity中当颜色值小0时会按0处理,所以我们最后的数学表达式为:diffuse = I*max(0,dot(N,L));

在Unity Shader

Shader "My/03_1 shader"{Properties{_Diffuse("Diffuse",Color) = (1,1,1,1)}SubShader{Pass{CGPROGRAM#include "Lighting.cginc"//引入灯光库#pragma vertex vert#pragma fragment fragfixed4 _Diffuse;//application to vertexstruct a2v{float4 vertex:POSITION;float3 normal:NORMAL;};struct v2f{float4 position:SV_POSITION;float3 color:COLOR;};v2f vert(a2v v){v2f f;f.position = UnityObjectToClipPos(v.vertex);//顶点转为裁剪空间fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.rgb;//环境光fixed3 light_dir = normalize(_WorldSpaceLightPos0.xyz);//归一化光源的方向fixed3 normal_dir = normalize(mul(v.normal,(float3x3)unity_WorldToObject));//先把法线方向转为世界空间,归一化法线方向//兰伯特光照模型,直射光颜色*max(0,cosƟ(光和法线的夹角)),cosƟ = 光源的方向(向量)点乘法线方向fixed3 diffuse = _LightColor0.rgb*max(0,dot(light_dir,normal_dir))*_Diffuse.xyz;//点成求cos,然后灯光颜色矩阵相乘获取漫反射光f.color = diffuse+ambient;return f;}fixed4 frag(v2f f):SV_Target{return fixed4(f.color,1);}ENDCG}}}

半兰伯特光照模型

把上面的效果应用到Unity中,可以看出不被光照射的背面是比较暗的,此时我们可以优化兰伯特光照模型,有人提出了半兰伯特光照模型。
由上面的公式diffuse = I*max(0,dot(N,L)),可知dot(N,L)的范围为-1~1,如果我们可以把范围调整为0 ~1,那么模型看起来会更亮些,我们优化下让 dot(L,N)*0.5+0.5,这样范围就改为了0 ~1。
最后的公式为:
diffuse = I*max(0,dot(N,L)*0.5+0.5)

半兰伯特Shader程序

Shader "My/05 shader"{SubShader{Pass{CGPROGRAM#include "Lighting.cginc"//引入灯光库#pragma vertex vert#pragma fragment frag//application to vertexstruct a2v{float4 vertex:POSITION;float3 normal:NORMAL;};struct v2f{float4 position:SV_POSITION;float3 color:COLOR;};v2f vert(a2v v){v2f f;f.position = UnityObjectToClipPos(v.vertex);//顶点转为裁剪空间fixed3 ambient = UNITY_LIGHTMODEL_AMBIENT.rgb;//环境光fixed3 light_dir = normalize(_WorldSpaceLightPos0.xyz);//归一化灯光方向fixed3 normal_dir = normalize(mul(v.normal,(float3x3)unity_WorldToObject));//先把法线方向转为世界空间,归一化法线方向//半兰伯特光照模型fixed3 diffuse = _LightColor0.rgb*max(0,(dot(light_dir,normal_dir)*0.5+0.5));//点成求cos,然后灯光颜色矩阵相乘获取漫反射光f.color = diffuse+ambient;return f;}fixed4 frag(v2f f):SV_Target{return fixed4(f.color,1);}ENDCG}}}

从相同的视角来观察两个模型,效果对比如下:
在这里插入图片描述

相关内容

热门资讯

监控摄像头接入GB28181平... 流程简介将监控摄像头的视频在网站和APP中直播,要解决的几个问题是:1&...
Windows10添加群晖磁盘... 在使用群晖NAS时,我们需要通过本地映射的方式把NAS映射成本地的一块磁盘使用。 通过...
protocol buffer... 目录 目录 什么是protocol buffer 1.protobuf 1.1安装  1.2使用...
Fluent中创建监测点 1 概述某些仿真问题,需要创建监测点,用于获取空间定点的数据࿰...
educoder数据结构与算法...                                                   ...
MySQL下载和安装(Wind... 前言:刚换了一台电脑,里面所有东西都需要重新配置,习惯了所...
MFC文件操作  MFC提供了一个文件操作的基类CFile,这个类提供了一个没有缓存的二进制格式的磁盘...
在Word、WPS中插入AxM... 引言 我最近需要写一些文章,在排版时发现AxMath插入的公式竟然会导致行间距异常&#...
有效的括号 一、题目 给定一个只包括 '(',')','{','}'...
【Ctfer训练计划】——(三... 作者名:Demo不是emo  主页面链接:主页传送门 创作初心ÿ...