光照模型:Phong模型

Phong模型

1) 向量定义(对应图中几何关系)

  • P:表面上一点(当前像素/片元的位置)
  • N:该点的单位法线(图中虚线方向),N = normalize(N)
  • L:光照方向(从点 P 指向光源 LightPos
    L = normalize(LightPos - P)
  • V:视线方向(从点 P 指向相机/眼睛 ViewPos,图中右上)
    V = normalize(ViewPos - P)
  • R:反射方向(镜面反射方向,用于高光)
    R = reflect(-L, N)
    说明:reflect(I, N)I 是"入射方向",这里入射光是从光源打到表面点,方向可写成 -L

2) Phong 三项:环境光 + 漫反射 + 镜面反射

Phong 模型把最终颜色/亮度写成: I = I_ambient + I_diffuse + I_specular

(1) 环境光 Ambient(不考虑方向,给整体一个底色)

  • I_ambient = Ka * Ia
  • Ka:材质环境系数(可用颜色 vec3
  • Ia:环境光强/环境光颜色(全局常量或场景参数)

(2) 漫反射 Diffuse(Lambert),取决于入射角(图中箭头与法线夹角)

  • 夹角余弦:NdotL = max(dot(N, L), 0)
  • I_diffuse = Kd * Il * NdotL
  • Kd:材质漫反射系数/底色
  • Il:光源颜色/强度

(3) 镜面反射 Specular(Phong 高光),取决于观察方向与反射方向的接近程度

  • RdotV = max(dot(R, V), 0)
  • I_specular = Ks * Il * (RdotV ^ shininess)
  • Ks:材质镜面系数(高光颜色/强度)
  • shininess:高光指数(越大高光越小越"硬")

常见细节:当 dot(N, L) <= 0 时,镜面项通常也置 0(背光面不产生高光):

  • if (NdotL <= 0) spec = 0

3) 点光源距离衰减(图里像是点光源,更真实)

若光源是点光源,强度随距离减弱:

  • d = length(LightPos - P)
  • atten = 1 / (kc + kl*d + kq*d*d)
  • I_diffuseI_specular 乘上 atten

4) 最终合成(按颜色通道 vec3 计算)

  • Color = Ka*Ia + atten * (Kd*Il*NdotL + Ks*Il*pow(RdotV, shininess))
  • 最后可 clamp(Color, 0, 1) 或做 HDR/色调映射(看你的渲染管线)
cpp 复制代码
#version 330 core
in vec3 vFragPos;
in vec3 vNormal;
out vec4 FragColor;

uniform vec3 uViewPos;
uniform vec3 uLightPos;
uniform vec3 uLightColor;
uniform vec3 uObjectColor;
uniform float uShininess; // 如 32.0

void main() {
    vec3 N = normalize(vNormal);
    vec3 L = normalize(uLightPos - vFragPos);
    vec3 V = normalize(uViewPos - vFragPos);
    vec3 R = reflect(-L, N);

    float diff = max(dot(N, L), 0.0);
    float spec = pow(max(dot(R, V), 0.0), uShininess);

    vec3 ambient  = 0.1 * uLightColor;
    vec3 diffuse  = diff * uLightColor;
    vec3 specular = 0.3 * spec * uLightColor;

    vec3 color = (ambient + diffuse + specular) * uObjectColor;
    FragColor = vec4(color, 1.0);
}

Blinn-Phong模型

在phong模型的基础上,改变高光的计算,不使用反射向量,取而代之的是半程向量HH = normalize(L + V)

几何意义:HLV 的角平分方向。当 N 越接近 H,越容易出现高光。

  • 镜面反射(替换为 Half Vector):
    I_specular = Ks * Il * (max(dot(N, H), 0) ^ shininess)

其他项计算和phong模型的计算一致。

cpp 复制代码
#version 330 core
in vec3 vFragPos;
in vec3 vNormal;
out vec4 FragColor;

uniform vec3 uViewPos;
uniform vec3 uLightPos;
uniform vec3 uLightColor;
uniform vec3 uObjectColor;
uniform float uShininess; // 如 32.0

void main() {
    vec3 N = normalize(vNormal);
    vec3 L = normalize(uLightPos - vFragPos);
    vec3 V = normalize(uViewPos - vFragPos);
    vec3 H = normalize(L + V);

    float diff = max(dot(N, L), 0.0);
    float spec = pow(max(dot(N, H), 0.0), uShininess);

    vec3 ambient  = 0.1 * uLightColor;
    vec3 diffuse  = diff * uLightColor;
    vec3 specular = 0.3 * spec * uLightColor;

    vec3 color = (ambient + diffuse + specular) * uObjectColor;
    FragColor = vec4(color, 1.0);
}


相关推荐
lclin_20201 天前
VS2010兼容|C++系统全能监控工具(彩色界面+日志带单位+完整版)
c++·windows·系统监控·vs2010·编程实战
paeamecium1 天前
【PAT甲级真题】- Cars on Campus (30)
数据结构·c++·算法·pat考试·pat
UrSpecial1 天前
从零实现C++轻量线程池
c++·线程池
chh5631 天前
C++--模版初阶
c语言·开发语言·c++·学习·算法
会编程的土豆1 天前
01背包与完全背包详解
开发语言·数据结构·c++·算法
hetao17338371 天前
2026-04-12~14 hetao1733837 的刷题记录
c++·算法
智者知已应修善业1 天前
【51单片机4位数循环小数位移数值位移】2023-6-9
c++·经验分享·笔记·算法·51单片机
王璐WL1 天前
【C++】string,vector和list对比
c++·list
不爱吃炸鸡柳1 天前
算法复杂度从入门到精通:时间与空间复杂度全解析
开发语言·c++·算法
拳里剑气1 天前
C++算法:二分查找
c++·算法·二分查找·学习方法