当前位置: 首页 > news >正文

OpenGL shader开发实战学习笔记:第十二章 深入光照

1. 深入光照

1.1. 平行光

我们在前面的章节中,已经介绍了平行光的基本原理和实现步骤

平行光的基本原理是,所有的光都从同一个方向照射到物体上,这个方向就是平行光的方向。

1.2. 点光源

点光源的基本原理是,所有的光都从一个点向各个方向照射到物体上,这个点就是点光源的位置。

1.2.1. 点光源的特点

点光源是计算机图形学和现实照明中常见的一种光源类型,它模拟了从一个点向各个方向均匀发射光线的光源。以下是点光源的主要特点:

1.2.1.1. 物理特性方面
  1. 位置决定性:点光源有明确的位置坐标,所有光线从该点向四面八方发射。在三维空间中,其位置可以用一个三维向量 (x, y, z) 来精确表示。在场景中移动点光源的位置,会直接改变物体受光的区域和强度分布。
  2. 光线发散性:光线从点光源出发,呈辐射状向周围空间发散传播。随着传播距离的增加,光线覆盖的面积会逐渐增大,这符合光的传播规律。
  3. 光照衰减:点光源的光照强度会随着距离的增加而减弱。根据平方反比定律,光照强度与距离的平方成反比,即距离点光源越远,物体接收到的光照越弱。
1.2.1.2. 渲染效果方面
  1. 产生阴影:由于点光源的光线是从一个点发出的,物体被其照射时会在背后产生明显的阴影。阴影的形状和范围取决于物体的形状、位置以及点光源的位置和强度。
  2. 多角度照明:能从多个角度照亮物体,使物体表面产生丰富的明暗变化,增强物体的立体感和层次感。物体朝向点光源的面会被照亮,而背向的面则处于阴影中。
  3. 局部照明效果:点光源通常用于突出场景中的特定物体或区域,营造局部照明效果。例如,在游戏场景中,可以用点光源模拟火把、吊灯等,使这些发光物体周围的环境更加生动。
1.2.1.3. 计算复杂度方面
  1. 计算量较大:与平行光等简单光源相比,点光源的光照计算更为复杂。因为需要考虑光源位置、物体与光源的距离、光照衰减等多个因素,所以在实时渲染中,大量使用点光源可能会对性能产生较大影响。
  2. 可优化性:为了提高渲染效率,有多种针对点光源的优化算法,如光照探针、光照贴图等。这些方法可以在一定程度上减少点光源的计算量,同时保持较好的光照效果。

1.2.2. 点光源的实现步骤

点光源的实现步骤如下:

  1. 计算点光源到物体表面的距离。
  2. 计算点光源到物体表面的方向向量。
  3. 计算点光源到物体表面的光照强度。
  4. 将光照强度应用到物体表面。
1.2.2.1. PointLightData

点光源的数据结构如下:

struct PointLightData {glm::vec3 position;glm::vec3 color;float intensity;float radius;
};
1.2.2.2. fragment shader

点光源到物体表面的距离可以通过以下公式计算:

    vec3 toLight=(lightPos-fragWorldPos);vec3 lightDir=normalize(toLight); // 从片元到光源的向量float dist=length(toLight); // 片元到光源的距离float fallOff=1.0-(dist/lightRadius); // 衰减fallOff=clamp(fallOff,0.0,1.0); // 限制在0到1之间vec3 lightColNew=lightCol*fallOff; // 应用衰减到光源颜色
1.2.2.3. 通过按键控制点光源

我们可以通过按键控制点光源的位置,以便观察点光源不同位置对于物体的影响,代码如下:

void ofApp::keyPressed(int key)
{if (key == 'a') {pointLightData.position.x -= 0.01f;}else if (key == 'd') {pointLightData.position.x += 0.01f;}else if (key == 'w') {pointLightData.position.y += 0.01f;}else if (key =='s') {pointLightData.position.y -= 0.01f;}else if (key == 'q') {pointLightData.position.z -= 0.01f;}else if (key == 'e') {pointLightData.position.z += 0.01f;}printf("pointLightData.position.x = %f , pointLightData.position.y = %f, pointLightData.position.z = %f\n", pointLightData.position.x, pointLightData.position.y, pointLightData.position.z);}

以下两张图分别是点光源位置不同时,对于物体的影响。

1.3. 聚光灯


聚光灯是一种特殊的光源,它从某个方向照射物体,并在物体表面产生锥形的光照区域。聚光灯的原理和实现步骤如下:

  1. 计算聚光灯到物体表面的距离。
  2. 计算聚光灯到物体表面的方向向量。
  3. 计算聚光灯到物体表面的光照强度。
  4. 计算聚光灯到物体表面的方向向量与聚光灯方向的角度,可以转换为余弦值。即:cosθ = dot(lightDir,lightDirToFragment)
  5. 判断聚光灯是否能照射到物体上,即cosθ是否大于聚光灯的lightCutoff值(角度<聚光灯的角度)。
  6. 将光照强度应用到物体表面。

以下是运行效果:
从中可以看到,聚光灯的光照区域是一个圆形的区域。

1.3.1. 核心代码(fragment shader)

    vec3 toLight = lightPos - fragWorldPos; vec3 lightDir = normalize(toLight);float angle = dot(lightConeDir, -lightDir);float falloff = 0.0;if (angle > lightCutoff){falloff = 1.0;}float diffAmt = diffuse(lightDir, nrm) * falloff;

1.3.2. 完整代码(fragment shader)

#version 410uniform vec3 lightCol;
uniform vec3 lightConeDir;
uniform vec3 lightPos;
uniform float lightCutoff; 
uniform vec3 cameraPos;
uniform vec3 ambientCol;
uniform sampler2D diffuseTex;
uniform sampler2D specTex;
uniform sampler2D normTex;
uniform samplerCube envMap;in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;
in mat3 TBN;
out vec4 outCol;float diffuse(vec3 lightDir, vec3 nrm)
{float diffAmt = max(0.0, dot(nrm, lightDir));return diffAmt;
}float specular(vec3 lightDir, vec3 viewDir, vec3 nrm, float shininess)
{vec3 halfVec = normalize(viewDir + lightDir);float specAmt = max(0.0, dot(halfVec, nrm));return pow(specAmt, shininess);
}void main(){	vec3 nrm = texture(normTex, fragUV).rgb;nrm = normalize(nrm * 2.0 - 1.0);   nrm = normalize(TBN * nrm); vec3 viewDir = normalize( cameraPos - fragWorldPos); vec3 envSample = texture(envMap, reflect(-viewDir, nrm)).xyz;vec3 sceneLight = mix(lightCol, envSample + lightCol * 0.5, 0.5);	vec3 toLight = lightPos - fragWorldPos; vec3 lightDir = normalize(toLight);float angle = dot(lightConeDir, -lightDir);float falloff = 0.0;if (angle > lightCutoff){falloff = 1.0;}float diffAmt = diffuse(lightDir, nrm) * falloff;float specAmt = specular(lightDir, viewDir, nrm, 4.0) * falloff;vec3 diffCol = texture(diffuseTex, fragUV).xyz * sceneLight * diffAmt;float specMask = texture(specTex, fragUV).x;vec3 specCol = specMask * sceneLight * specAmt;outCol = vec4(diffCol + specCol + ambientCol, 1.0);}

1.4. 多个灯源(shader中固定实现)

运行效果:

采用在shader中固定实现,这种方式我们不推荐,因为这样会使得shader代码变得复杂,而且效率不高。但这种方式可以让我们了解,多个光源是如何在shader中实现的。

1.4.1. 实现步骤

  1. 定义多个光源的数据结构。
  2. 在fragment shader中,遍历所有的光源,计算每个光源对物体的光照强度。
  3. 将所有光源对物体的光照强度相加,得到最终的光照强度。

1.4.2. 核心代码(fragment shader)

vec3 finalColor = vec3(0,0,0);
//遍历所有的平行光源
for (int i = 0; i < NUM_DIR_LIGHTS; ++i) {DirectionalLight light = directionalLights[i];vec3 sceneLight = mix(light.color, envSample + light.color * 0.5, 0.5);	float diffAmt = diffuse(light.direction, nrm);float specAmt = specular(light.direction, viewDir, nrm, 4.0) * specMask;vec3 specCol = specMask * sceneLight * specAmt;finalColor += diffuseColor * diffAmt * sceneLight;finalColor += specCol * light.color;}
//遍历所有的点光源
//...

1.4.3. 完整代码(fragment shader)

#version 410struct DirectionalLight {vec3 direction;vec3 color;
};  struct PointLight {    vec3 position;vec3 color;float radius;
};  struct SpotLight{vec3 position;vec3 direction;vec3 color;float cutoff;
};#define NUM_DIR_LIGHTS 1
#define NUM_POINT_LIGHTS 2
#define NUM_SPOT_LIGHTS 2uniform DirectionalLight directionalLights[NUM_DIR_LIGHTS];
uniform PointLight pointLights[NUM_POINT_LIGHTS];
uniform SpotLight spotLights[NUM_SPOT_LIGHTS];
uniform sampler2D diffuseTex;
uniform sampler2D specTex;
uniform sampler2D normTex;
uniform samplerCube envMap;
uniform vec3 cameraPos;
uniform vec3 ambientCol;in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;
in mat3 TBN;out vec4 outCol;float diffuse(vec3 lightDir, vec3 nrm)
{float diffAmt = max(0.0, dot(nrm, lightDir));return diffAmt;
}float specular(vec3 lightDir, vec3 viewDir, vec3 nrm, float shininess)
{vec3 halfVec = normalize(viewDir + lightDir);float specAmt = max(0.0, dot(halfVec, nrm));return pow(specAmt, shininess);
}void main(){	vec3 nrm = texture(normTex, fragUV).rgb;nrm = normalize(nrm * 2.0 - 1.0);   nrm = normalize(TBN * nrm); vec3 viewDir = normalize( cameraPos - fragWorldPos); 	vec3 envSample = texture(envMap, reflect(-viewDir, nrm)).xyz;float specMask = texture(specTex, fragUV).x;vec3 diffuseColor = texture(diffuseTex, fragUV).xyz;vec3 finalColor = vec3(0,0,0);for (int i = 0; i < NUM_DIR_LIGHTS; ++i) {DirectionalLight light = directionalLights[i];vec3 sceneLight = mix(light.color, envSample + light.color * 0.5, 0.5);	float diffAmt = diffuse(light.direction, nrm);float specAmt = specular(light.direction, viewDir, nrm, 4.0) * specMask;vec3 specCol = specMask * sceneLight * specAmt;finalColor += diffuseColor * diffAmt * sceneLight;finalColor += specCol * light.color;}for (int i = 0; i < NUM_POINT_LIGHTS; ++i){PointLight light = pointLights[i];vec3 sceneLight = mix(light.color, envSample + light.color * 0.5, 0.5);	vec3 toLight = light.position - fragWorldPos; vec3 lightDir = normalize(toLight);float distToLight = length(toLight); float falloff = 1.0 - (distToLight / light.radius);float diffAmt = diffuse(lightDir, nrm) * falloff;float specAmt = specular(lightDir, viewDir, nrm, 4.0) * specMask * falloff;vec3 specCol = specMask * sceneLight * specAmt;finalColor += diffAmt * sceneLight * diffuseColor;finalColor += specCol;}for (int i = 0; i < NUM_SPOT_LIGHTS; ++i){SpotLight light = spotLights[i];vec3 sceneLight = mix(light.color, envSample + light.color * 0.5, 0.5);	vec3 toLight = light.position - fragWorldPos; vec3 lightDir = normalize(toLight);float angle = dot(light.direction, -lightDir);float falloff = (angle > light.cutoff) ? 1.0 : 0.0;float diffAmt = diffuse(lightDir, nrm) * falloff;float specAmt = specular(lightDir, viewDir, nrm, 4.0) * specMask * falloff;vec3 specCol = specMask * sceneLight * specAmt;finalColor += diffAmt * sceneLight * diffuseColor;finalColor += specCol;}outCol = vec4(finalColor + ambientCol, 1.0);
}

1.5. Multi-pass(多通道)

在着色器编程里,Multi-pass(多通道)着色是一种渲染技术,它将一个复杂的渲染任务拆分成多个连续的渲染通道(pass)来完成。每个通道都有特定的渲染目标和操作,最终把各个通道的结果组合起来,实现复杂的视觉效果。下面从原理、工作流程、应用场景和优缺点几个方面详细介绍。

1.5.1. 原理

Multi-pass 着色的核心思想是把复杂的渲染效果分解为多个简单的步骤,每个步骤对应一个渲染通道。在每个通道中,GPU 会对场景或者特定对象进行渲染,将结果存储在帧缓冲(Framebuffer)或者纹理中,后续的通道可以利用这些中间结果进行进一步处理,最终得到完整的渲染效果。

1.5.2. 工作流程

  1. 初始化 :设置好初始的渲染状态,比如视口大小、清除颜色缓冲区和深度缓冲区等。
  2. 通道渲染 :依次执行各个渲染通道,每个通道都有自己的着色器程序、渲染目标和渲染参数。在每个通道结束后,可能需要保存渲染结果到纹理或者帧缓冲中。
  3. 结果合并 :最后一个通道或者后续处理步骤会把前面各个通道的结果进行合并,生成最终的渲染图像。

1.5.3. 优缺点

1.5.3.1. 优点
  • 实现复杂效果 :能够实现单通道难以完成的复杂视觉效果,提升渲染质量。
  • 模块化设计 :将渲染任务分解为多个通道,每个通道的逻辑相对简单,便于开发和维护。

1.5.4. 缺点

  • 性能开销大 :每个通道都需要进行一次完整的渲染过程,会增加 GPU 的计算量和内存开销,降低渲染性能。
  • 资源占用多 :需要额外的帧缓冲和纹理来存储中间结果,增加了内存资源的占用。

1.5.5. 概念

1.5.5.1. glDepthFunc(GL_LEQUAL)

glDepthFunc(GL_LEQUAL); 是 OpenGL 里用于设置深度测试函数的调用语句。深度测试是渲染管线中非常关键的一部分,它依据片段的深度值来决定新片段是否能覆盖帧缓冲区里已有的片段。

函数解释

  • glDepthFunc 是 OpenGL 的一个函数,作用是指定深度比较函数。该函数会根据新片段和深度缓冲区中已存片段的深度值,判断新片段是否可以替换掉已有的片段。
  • GL_LEQUAL 是众多深度比较函数中的一种,代表“小于或等于”。当调用 glDepthFunc(GL_LEQUAL) 后,若新传入片段的深度值小于或等于深度缓冲区中存储的深度值,深度测试就会通过,这个新片段就可能会被绘制到屏幕上;反之,若不满足条件,该片段就会被丢弃。
  • GL_LESS 若新片段的深度值小于深度缓冲区中存储的深度值,深度测试通过。

调用glDepthFunc(GL_LEQUAL) 函数后,就不会忽略片元被绘制到前面通道已经绘制的同一个点上

1.5.6. 代码

1.5.6.1. 核心代码(ofApp.cpp)
void ofApp::beginRenderingPointLights()
{ofEnableAlphaBlending();ofEnableBlendMode(ofBlendMode::OF_BLENDMODE_ADD);glDepthFunc(GL_LEQUAL);
}void ofApp::endRenderingPointLights()
{ofDisableAlphaBlending();ofDisableBlendMode();glDepthFunc(GL_LESS);
}
void ofApp::draw() {using namespace glm;cam.pos = glm::vec3(0, 0.75f, 1.0);mat4 proj = perspective(cam.fov, 1024.0f / 768.0f, 0.01f, 10.0f);mat4 view = inverse(translate(cam.pos));drawSkybox(proj, view);drawWater(dirLight, proj, view);drawShield(dirLight, proj, view);beginRenderingPointLights();for (int i = 0; i < pointLights.size(); ++i){drawWater(pointLights[i], proj, view);drawShield(pointLights[i], proj, view);}endRenderingPointLights();}
1.5.6.2. 完整代码(ofApp.cpp)
#include "ofApp.h"
#include <vector>
#include "ofMainLoop.h"
//--------------------------------------------------------------void calcTangents(ofMesh& mesh)
{using namespace glm;std::vector<vec4> tangents;tangents.resize(mesh.getNumVertices());uint indexCount = mesh.getNumIndices();const vec3* vertices = mesh.getVerticesPointer();const vec2* uvs = mesh.getTexCoordsPointer();const uint* indices = mesh.getIndexPointer();for (uint i = 0; i < indexCount - 2; i += 3){const vec3& v0 = vertices[indices[i]];const vec3& v1 = vertices[indices[i + 1]];const vec3& v2 = vertices[indices[i + 2]];const vec2& uv0 = uvs[indices[i]];const vec2& uv1 = uvs[indices[i + 1]];const vec2& uv2 = uvs[indices[i + 2]];vec3 edge1 = v1 - v0;vec3 edge2 = v2 - v0;vec2 dUV1 = uv1 - uv0;vec2 dUV2 = uv2 - uv0;float f = 1.0f / (dUV1.x * dUV2.y - dUV2.x * dUV1.y);vec4 tan;tan.x = f * (dUV2.y * edge1.x - dUV1.y * edge2.x);tan.y = f * (dUV2.y * edge1.y - dUV1.y * edge2.y);tan.z = f * (dUV2.y * edge1.z - dUV1.y * edge2.z);tan.w = 0;tan = normalize(tan);tangents[indices[i]] += (tan);tangents[indices[i + 1]] += (tan);tangents[indices[i + 2]] += (tan);}int numColors = mesh.getNumColors();for (int i = 0; i < tangents.size(); ++i){vec3 t = normalize(tangents[i]);if (i >= numColors){mesh.addColor(ofFloatColor(t.x, t.y, t.z, 0.0));}else{mesh.setColor(i, ofFloatColor(t.x, t.y, t.z, 0.0));}}
}void ofApp::setup() {ofSetVerticalSync(false);ofDisableArbTex();ofEnableDepthTest();planeMesh.load("plane.ply");shieldMesh.load("shield.ply");cubeMesh.load("cube.ply");dirLightShaders[0].load("mesh.vert", "dirLight.frag");pointLightShaders[0].load("mesh.vert", "pointLight.frag");dirLightShaders[1].load("water.vert", "dirLightWater.frag");pointLightShaders[1].load("water.vert", "pointLightWater.frag");diffuseTex.load("shield_diffuse.png");specTex.load("shield_spec.png");nrmTex.load("shield_normal.png");waterNrm.load("water_nrm.png");waterNrm.getTexture().enableMipmap();waterNrm.getTexture().generateMipmap();waterNrm.getTexture().setTextureMinMagFilter(GL_LINEAR_MIPMAP_LINEAR, GL_LINEAR);waterNrm.getTexture().setTextureWrap(GL_REPEAT, GL_REPEAT);cam.pos = glm::vec3(0, 0.75f, 1.0f);cam.fov = glm::radians(90.0f);calcTangents(shieldMesh);calcTangents(planeMesh);skyboxShader.load("skybox.vert", "skybox.frag");cam.pos = glm::vec3(0, 0.75f, 1.0f);cam.fov = glm::radians(90.0f);cubemap.load("night_front.jpg", "night_back.jpg","night_right.jpg", "night_left.jpg","night_top.jpg", "night_bottom.jpg");PointLight pl0;pl0.color = glm::vec3(1, 0, 0);pl0.radius = 1.0f;pl0.position = glm::vec3(-0.5, 0.35, 0.25);pl0.intensity = 3.0;PointLight pl1;pl1.color = glm::vec3(0, 1, 0);pl1.radius = 1.0f;pl1.position = glm::vec3(0.5, 0.35, 0.25);pl1.intensity = 3.0;PointLight pl2;pl2.color = glm::vec3(0, 0, 1);pl2.radius = 1.0f;pl2.position = glm::vec3(0.0, 0.7, 0.25);pl2.intensity = 3.0;pointLights.push_back(pl0);pointLights.push_back(pl1);pointLights.push_back(pl2);dirLight.color = glm::vec3(1, 1, 0);dirLight.intensity = 0.25f;dirLight.direction = glm::vec3(0, 0, -1);}glm::vec3 getLightDirection(DirectionalLight& l)
{return glm::normalize(l.direction * -1.0f);
}glm::vec3 getLightColor(DirectionalLight& l)
{return l.color * l.intensity;
}glm::vec3 getLightColor(PointLight& l)
{return l.color * l.intensity;
}//--------------------------------------------------------------
void ofApp::update()
{
}void ofApp::drawWater(Light& light, glm::mat4& proj, glm::mat4& view)
{using namespace glm;static float t = 0.0f;t += ofGetLastFrameTime();vec3 right = vec3(1, 0, 0);mat4 rotation = rotate(radians(-90.0f), right);mat4 model = rotation * scale(vec3(10.0, 10.0, 10.0));mat4 mvp = proj * view * model;mat3 normalMatrix = mat3(transpose(inverse(model)));ofShader shd = light.isPointLight() ? pointLightShaders[1] : dirLightShaders[1];shd.begin();light.apply(shd);shd.setUniformMatrix4f("mvp", mvp);shd.setUniformMatrix4f("model", model);shd.setUniformMatrix3f("normal", normalMatrix);shd.setUniform3f("meshSpecCol", glm::vec3(1, 1, 1));shd.setUniformTexture("normTex", waterNrm, 0);;shd.setUniformTexture("envMap", cubemap.getTexture(), 1);shd.setUniform1f("time", t);shd.setUniform3f("ambientCol", glm::vec3(0.0, 0.0, 0.0));shd.setUniform3f("cameraPos", cam.pos);planeMesh.draw();shd.end();
}void ofApp::drawSkybox(glm::mat4& proj, glm::mat4& view)
{using namespace glm;mat4 model = translate(cam.pos);mat4 mvp = proj * view * model;ofShader& shd = skyboxShader;glDepthFunc(GL_LEQUAL);shd.begin();shd.setUniformMatrix4f("mvp", mvp);shd.setUniformTexture("envMap", cubemap.getTexture(), 0);cubeMesh.draw();shd.end();glDepthFunc(GL_LESS);
}void ofApp::drawShield(Light& light, glm::mat4& proj, glm::mat4& view)
{using namespace glm;mat4 model = translate(vec3(0.0, 0.75, 0.0f));mat4 mvp = proj * view * model;mat3 normalMatrix = mat3(transpose(inverse(model)));ofShader shd = light.isPointLight() ? pointLightShaders[0] : dirLightShaders[0];shd.begin();light.apply(shd);shd.setUniformMatrix4f("mvp", mvp);shd.setUniformMatrix4f("model", model);shd.setUniformMatrix3f("normal", normalMatrix);shd.setUniform3f("meshSpecCol", glm::vec3(1, 1, 1));shd.setUniformTexture("diffuseTex", diffuseTex, 0);;shd.setUniformTexture("specTex", specTex, 1);shd.setUniformTexture("normTex", nrmTex, 2);shd.setUniformTexture("envMap", cubemap.getTexture(), 3);shd.setUniform3f("ambientCol", glm::vec3(0.0, 0.0, 0.0));shd.setUniform3f("cameraPos", cam.pos);shieldMesh.draw();shd.end();
}void ofApp::beginRenderingPointLights()
{ofEnableAlphaBlending();ofEnableBlendMode(ofBlendMode::OF_BLENDMODE_ADD);glDepthFunc(GL_LEQUAL);
}void ofApp::endRenderingPointLights()
{ofDisableAlphaBlending();ofDisableBlendMode();glDepthFunc(GL_LESS);
}//--------------------------------------------------------------
void ofApp::draw() {using namespace glm;cam.pos = glm::vec3(0, 0.75f, 1.0);mat4 proj = perspective(cam.fov, 1024.0f / 768.0f, 0.01f, 10.0f);mat4 view = inverse(translate(cam.pos));drawSkybox(proj, view);drawWater(dirLight, proj, view);drawShield(dirLight, proj, view);beginRenderingPointLights();for (int i = 0; i < pointLights.size(); ++i){drawWater(pointLights[i], proj, view);drawShield(pointLights[i], proj, view);}endRenderingPointLights();}
1.5.6.3. dirLight.frag
#version 410uniform vec3 lightDir;
uniform vec3 lightCol;
uniform vec3 cameraPos;
uniform vec3 ambientCol;
uniform sampler2D diffuseTex;
uniform sampler2D specTex;
uniform sampler2D normTex;
uniform samplerCube envMap;
in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;
in mat3 TBN;out vec4 outCol;void main()
{vec3 nrm = texture(normTex, fragUV).rgb;nrm = normalize(nrm * 2.0 - 1.0);   nrm = normalize(TBN * nrm); vec3 diffCol = vec3(0,0,0);vec3 specCol = vec3(0,0,0);vec3 viewDir = normalize( cameraPos - fragWorldPos); 	vec3 halfVec = normalize(viewDir + lightDir);vec3 envSample = texture(envMap, reflect(-viewDir, nrm)).xyz;vec3 sceneLight = mix(lightCol, envSample + lightCol * 0.5, 0.5);float diffAmt = max(0.0, dot(nrm, lightDir));diffCol += texture(diffuseTex, fragUV + nrm.xy).xyz * sceneLight * diffAmt;float specAmt = max(0.0, dot(halfVec, nrm));float specBright = pow(specAmt, 4.0);specCol += texture(specTex, fragUV + nrm.xy).x * sceneLight * specBright;outCol = vec4(diffCol + specCol + ambientCol, 1.0);}
1.5.6.4. dirLightWater.frag
#version 410uniform vec3 lightDir;
uniform vec3 lightCol;
uniform vec3 cameraPos;
uniform vec3 ambientCol;
uniform sampler2D normTex;
uniform samplerCube envMap;
in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;in mat3 TBN;out vec4 outCol;in vec2 fragUV2;
void main()
{vec3 nrm = texture(normTex, fragUV).rgb;nrm = (nrm * 2.0 - 1.0);   vec3 nrm2 = texture(normTex, fragUV2).rgb;nrm2 = (nrm2 * 2.0 - 1.0);   nrm = normalize(TBN * (nrm + nrm2)); vec3 viewDir = normalize( cameraPos - fragWorldPos); 	vec3 halfVec = normalize(viewDir + lightDir);vec3 envSample = texture(envMap, reflect(-viewDir, nrm)).xyz;vec3 sceneLight = mix(lightCol, envSample + lightCol * 0.5, 0.5);float diffAmt = max(0.0, dot(nrm, lightDir));vec3 diffCol = sceneLight * diffAmt;float specAmt = max(0.0, dot(halfVec, nrm));float specBright = pow(specAmt, 512.0);vec3 specCol = sceneLight * specBright;outCol = vec4(diffCol + specCol + ambientCol, 1.0); 
}
1.5.6.5. pointLight.frag
#version 410 uniform vec3 meshCol;
uniform vec3 lightPos; 
uniform float lightRadius;
uniform vec3 lightCol;
uniform vec3 cameraPos;
uniform vec3 ambientCol;
uniform sampler2D diffuseTex;
uniform sampler2D specTex;
uniform sampler2D normTex;
uniform samplerCube envMap;in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;
in mat3 TBN;
out vec4 outCol;float diffuse(vec3 lightDir, vec3 nrm)
{float diffAmt = max(0.0, dot(nrm, lightDir));return diffAmt;
}float specular(vec3 lightDir, vec3 viewDir, vec3 nrm, float shininess)
{vec3 halfVec = normalize(viewDir + lightDir);float specAmt = max(0.0, dot(halfVec, nrm));return pow(specAmt, shininess);
}void main()
{	vec3 nrm = texture(normTex, fragUV).rgb;nrm = normalize(nrm * 2.0 - 1.0);   nrm = normalize(TBN * nrm); vec3 viewDir = normalize( cameraPos - fragWorldPos); 	vec3 envSample = texture(envMap, reflect(-viewDir, nrm)).xyz;vec3 sceneLight = mix(lightCol, envSample + lightCol * 0.5, 0.5);//manual light direction calculationvec3 toLight = lightPos - fragWorldPos; vec3 lightDir = normalize(toLight);float distToLight = length(toLight); float falloff = 1.0 - (distToLight / lightRadius);float diffAmt = diffuse(lightDir, nrm) * falloff; float specAmt = specular(lightDir, viewDir, nrm, 4.0) * falloff; vec3 diffCol = texture(diffuseTex, fragUV).xyz * sceneLight * diffAmt;float specMask = texture(specTex, fragUV).x;vec3 specCol = specMask * sceneLight * specAmt;outCol = vec4(diffCol + specCol, 1.0);
}
1.5.6.6. pointLightWater.frag
#version 410uniform vec3 lightPos; 
uniform float lightRadius;
uniform vec3 lightCol;
uniform vec3 cameraPos;
uniform vec3 ambientCol;
uniform sampler2D normTex;
uniform samplerCube envMap;in vec3 fragNrm;
in vec3 fragWorldPos;
in vec2 fragUV;
in vec2 fragUV2;
in mat3 TBN;out vec4 outCol;float diffuse(vec3 lightDir, vec3 nrm)
{float diffAmt = max(0.0, dot(nrm, lightDir));return diffAmt;
}float specular(vec3 lightDir, vec3 viewDir, vec3 nrm, float shininess)
{vec3 halfVec = normalize(viewDir + lightDir);float specAmt = max(0.0, dot(halfVec, nrm));return pow(specAmt, shininess);
}void main()
{vec3 nrm = texture(normTex, fragUV).rgb;nrm = (nrm * 2.0 - 1.0);   vec3 nrm2 = texture(normTex, fragUV2).rgb;nrm2 = (nrm2 * 2.0 - 1.0);   nrm = normalize(TBN * (nrm + nrm2)); //manual light direction calculationvec3 toLight = lightPos - fragWorldPos; vec3 lightDir = normalize(toLight);float distToLight = length(toLight); float falloff = 1.0 - (distToLight / lightRadius);vec3 viewDir = normalize( cameraPos - fragWorldPos); 	float diffAmt = diffuse(lightDir, nrm) * falloff; float specAmt = specular(lightDir, viewDir, nrm, 512.0) * falloff; vec3 diffCol = texture(envMap, (reflect(-viewDir, nrm))).xyz * lightCol * diffAmt;vec3 specCol = lightCol * specAmt;outCol = vec4(diffCol + specCol, 1.0); 
}

相关文章:

OpenGL shader开发实战学习笔记:第十二章 深入光照

1. 深入光照 1.1. 平行光 我们在前面的章节中&#xff0c;已经介绍了平行光的基本原理和实现步骤 平行光的基本原理是&#xff0c;所有的光都从同一个方向照射到物体上&#xff0c;这个方向就是平行光的方向。 1.2. 点光源 点光源的基本原理是&#xff0c;所有的光都从一个…...

1-1 什么是数据结构

1.0 数据结构的基本概念 数据结构是计算机科学中一个非常重要的概念&#xff0c;它是指在计算机中组织、管理和存储数据的方式&#xff0c;以便能够高效地访问和修改数据。简而言之&#xff0c;数据结构是用来处理数据的格式&#xff0c;使得数据可以被更有效地使用。 数据结构…...

【MySQL】:数据库事务管理

一&#xff1a;学习路径 &#xff08;1&#xff09;下载安装mysql &#xff08;2&#xff09;学习语言&#xff1a;SQL(操作数据库&#xff09; &#xff08;3&#xff09;mysql集群&#xff08;提升数据库存储效率&#xff09; &#xff08;4&#xff09;SQL使用&#xff0c;M…...

leetcode 647. Palindromic Substrings

题目描述 代码&#xff1a; class Solution { public:int countSubstrings(string s) {int n s.size();//i<j,dp[i][j]表示子字符串s[i,j]是否是回文子串,i>j的dp[i][j]不定义vector<vector<int>> dp(n,vector<int>(n,false));int res 0;for(int i …...

Linux-scp命令

scp&#xff08;Secure Copy Protocol&#xff09;是基于 SSH 的安全文件传输命令&#xff0c;用于在本地和远程主机之间加密传输文件或目录。以下是详细用法和示例&#xff1a; 基本语法 scp [选项] 源文件 目标路径常用选项 选项描述-P 端口号指定 SSH 端口&#xff08;默认…...

在CSDN的1095天(创作纪念日)

一早上收到CSDN官方的私信&#xff0c;时间飞逝&#xff0c;转眼间3年了…… 一些碎碎念… 算起来也断更一年多了&#xff0c;上一次更博客是去年的3月份&#xff0c;那时候还在实习&#xff0c;同时也是去年的三月份结束了第一段实习回学校准备考研&#xff0c;考完研12月开始…...

STM32——新建工程并使用寄存器以及库函数进行点灯

本文是根据江协科技提供的教学视频所写&#xff0c;旨在便于日后复习&#xff0c;同时供学习嵌入式的朋友们参考&#xff0c;文中涉及到的所有资料也均来源于江协科技&#xff08;资料下载&#xff09;。 新建工程并使用寄存器以及库函数进行点灯操作 新建工程步骤1.建立工程2.…...

【Linux】多线程任务模块

创建多个线程&#xff0c;同时完成任务 task.c #include <sys/types.h> #include <unistd.h> #include<stdio.h> #include <sys/wait.h> int create_process_tasks(Task_fun_t tasks[],int tsak_cnt) {pid_t pid;int i 0;for(i 0;i < 4;i){pid …...

Maxscript调用Newtonsoft.Json解析Json

Maxscript调用Newtonsoft.Json解析Json_newtonsoft.json maxscript-CSDN博客...

【前端】【面试】【业务场景】前端如何获取并生成设备唯一标识

✅ 总结 问题&#xff1a;前端如何获取并生成设备唯一标识&#xff1f; 核心要点&#xff1a;浏览器原生信息有限&#xff0c;但通过组合多个维度可生成设备指纹&#xff08;Device Fingerprint&#xff09;&#xff0c;用于唯一标识设备。 常见方式&#xff1a; 浏览器信息&…...

《Java面试通关宝典:基础篇》——Java面试题系列(持续更新)

《Java面试通关宝典&#xff1a;基础篇》是一篇针对Java编程初学者的面试宝典&#xff0c;旨在帮助大家快速复习Java编程语言的基础知识&#xff0c;提高面试竞争力。本文详细介绍了Java基础知识的各个方面&#xff0c;包括语言基础、面向对象、集合框架、异常处理等内容。同时…...

学习笔记(C++篇)--- Day 3

1.析构函数 析构函数不是完成对对象本身的销毁&#xff0c;C规定对象在销毁时会自动调用析构函数&#xff0c;完成对象中资源的清理释放工作。&#xff08;严格说&#xff0c;Date是不要析构函数的&#xff09; 特点&#xff1a; ①析构函数名是在类名钱加上字符~。 ②无参数&a…...

消息队列知识点详解

消息队列场景 什么是消息队列 可以把消息队列理解一个使用队列来通信的组件&#xff0c;它的本质是交换机队列的模式&#xff0c;实现发送消息&#xff0c;存储消息&#xff0c;消费消息的过程。 我们通常说的消息队列&#xff0c;MQ其实就是消息中间件&#xff0c;业界中比较…...

AI 赋能 3D 创作!Tripo3D 全功能深度解析与实操教程

大家好&#xff0c;欢迎来到本期科技工具分享&#xff01; 今天要给大家带来一款革命性的 AI 3D 模型生成平台 ——Tripo3D。 无论你是游戏开发者、设计师&#xff0c;还是 3D 建模爱好者&#xff0c;只要想降低创作门槛、提升效率&#xff0c;这款工具都值得深入了解。 接下…...

DeepSeek赋能Nuclei:打造网络安全检测的“超级助手”

引言 各位少侠&#xff0c;周末快乐&#xff0c;幸会幸会&#xff01; 今天唠一个超酷的技术组合——用AI大模型给Nuclei开挂&#xff0c;提升漏洞检测能力&#xff01; 想象一下&#xff0c;当出现新漏洞时&#xff0c;少侠们经常需要根据Nuclei模板&#xff0c;手动扒漏洞文章…...

【MySQL】表的约束(主键、唯一键、外键等约束类型详解)、表的设计

目录 1.数据库约束 1.1 约束类型 1.2 null约束 — not null 1.3 unique — 唯一约束 1.4 default — 设置默认值 1.5 primary key — 主键约束 自增主键 自增主键的局限性&#xff1a;经典面试问题&#xff08;进阶问题&#xff09; 1.6 foreign key — 外键约束 1.7…...

学习深度学习是否要先学习机器学习?工程师的路径选择策略

深度学习与机器学习的关系&#xff0c;如同摩天大楼与地基——前者是后者的高阶延伸&#xff0c;但能否绕过地基直接造楼&#xff1f;本文从技术本质、学习曲线、应用场景三个维度剖析这一关键问题。 一、技术血脉的承继关系 概念体系同源&#xff1a; 损失函数、梯度下降、过拟…...

高防服务器适合哪些行业使用

在当今数字化的时代&#xff0c;网络安全就如同城堡的城墙&#xff0c;而高防服务器则是这道城墙中的坚固堡垒。那么&#xff0c;究竟哪些行业特别需要高防服务器这位“守护天使”的庇佑呢&#xff1f; 首先&#xff0c;金融行业绝对是高防服务器的“头号粉丝”。想象一下&…...

【Docker-16】Docker Volume存储卷

Docker Volume(存储卷) 概念比喻镜像程序的光盘&#xff08;安装包&#xff09;容器安装并运行后的 App卷独立的文件夹或硬盘&#xff0c;用来保存数据宿主机装着 Docker 的电脑或服务器&#xff0c;是一切的基础 一、什么是存储卷? 存储卷就是将宿主机的本地文件系统中存在…...

后端如何生成验证码

目录 &#x1f510; 一、验证码类型与用途 &#x1f4f8; 二、图形验证码的原理&#xff08;Image Captcha&#xff09; &#x1f527; 核心流程 &#x1f6e0;️ 示例&#xff1a;用 Python Pillow 生成图形验证码 &#x1f4f1; 三、数字验证码&#xff08;短信/邮箱&a…...

微服务架构下数据库范式的失效与反范式设计的崛起

在传统单体应用中&#xff0c;关系型数据库范式设计被认为是数据库建模的黄金标准。然而&#xff0c;随着企业架构向分布式系统&#xff0c;特别是微服务架构演进&#xff0c;范式化数据库设计的有效性和适应性正受到前所未有的挑战。本文将深入剖析范式设计的设计哲学&#xf…...

Redis专题

前言 一&#xff1a;看到你的简历上写了你的项目里面用到了redis&#xff0c;为啥用redis&#xff1f; 因为传统的关系型数据库如Mysql,已经不能适用所有的场景&#xff0c;比如秒杀的库存扣减&#xff0c;APP首页的访问流量高峰等&#xff0c;都很容易把数据库打崩&#xff0…...

Vue的模板语法——指令语法

2025/4/21 向全栈工程师迈进&#xff01; 一、插值语法 之前通过插值语法&#xff0c;其显示的效果如下。 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widt…...

【Java面试笔记:基础】1.谈谈你对Java平台的理解?

前言 Java 是历史悠久且主流的编程语言&#xff0c;拥有庞大的开发者群体和广泛的应用领域。通过系统学习和实践&#xff0c;构建扎实的 Java 知识体系&#xff0c;提升面试成功率 笔记核心内容 1. Java 平台的核心特性 跨平台特性&#xff1a;Java 的核心特性之一是“Writ…...

NestJS-Knife4j

文章目录 前言✅ 一、什么是 Knife4j&#xff1f;✅ 二、Knife4j 与 Swagger 对比✅ 三、NestJS-Knife4j 集成1. 安装依赖2. 配置 Swagger 与 Knife4j3. 启动应用并访问接口文档 ✅ 四、功能增强1. **接口分组**2. **请求/响应示例**3. **接口文档的美化** ✅ 五、总结 前言 N…...

Linux环境准备(安装VirtualBox和Ubuntu,安装MySQL,MySQL启动、重启和停止)

目录 安装VirtualBox和Ubuntu 安装VirtualBox 安装Ubuntu 下载Ubuntu操作系统的镜像文件 创建虚拟机 虚拟机设置 启动虚拟机&#xff0c;安装Ubuntu系统 Ubuntu基础设置 设置系统为中文 设置中文输入法为拼音 修改分辨率 设置缩放比例 设置息屏时间 设置root用户…...

Nebula图数据库

Nebula 通常指 Nebula Graph&#xff0c;是一款开源的分布式图数据库系统1。以下是其相关介绍1&#xff1a; 特点 高性能&#xff1a;能处理千亿顶点和万亿边的超大规模数据集&#xff0c;提供毫秒级查询延迟&#xff0c;在处理大规模数据时&#xff0c;也能维持低时延的读写和…...

基于 Vue,使用Vuex 或事件总线实现跨组件通信

在基于 Vue Element UI 的 RuoYi 系统中&#xff0c;如果需要在一个界面&#xff08;界面 A&#xff09;执行某个操作后&#xff0c;通知另一个界面&#xff08;界面 B&#xff09;刷新&#xff0c;可以通过 Vuex 或事件总线&#xff08;Event Bus&#xff09;来实现跨组件通信…...

【C++篇】string类的终章:深浅拷贝 + 模拟实现string类的深度解析(附源码)

&#x1f4ac; 欢迎讨论&#xff1a;在阅读过程中有任何疑问&#xff0c;欢迎在评论区留言&#xff0c;我们一起交流学习&#xff01; &#x1f44d; 点赞、收藏与分享&#xff1a;如果你觉得这篇文章对你有帮助&#xff0c;记得点赞、收藏&#xff0c;并分享给更多对C感兴趣的…...

线性DP:最短编辑距离

Dp 状态表示 f&#xff08;i&#xff0c;j&#xff09; 集合所有将A[1~i]变成B[1~j]的操作方式属性min 状态计算 &#xff08;划分&#xff09; 增f(i,j)f(i,j-1)1//A[i]元素要增加&#xff0c;说明A前i位置与B前j-1相同删f(i,j)f(i-1,j)1//A[i]元素要删除&#xff0c;说明A前i…...

【全网最全】23种设计模式思维导图详解 | 含React/Vue/Spring实战案例

【全网最全】23种设计模式思维导图详解 | 含React/Vue/Spring实战案例 导图概述 本文通过高清思维导图系统梳理了23种设计模式&#xff0c;分为创建型、结构型、行为型三大类&#xff0c;并标注了各模式在主流框架&#xff08;如React、Vue、Spring&#xff09;中的典型应用场…...

8086微机原理与接口技术复习(1)存储器(2)接口

8086微机原理与接口技术复习&#xff08;1&#xff09;存储器&#xff08;2&#xff09;接口 存储器8086的存储空间存储器的拓展存储器的分类 接口8086I/O82558253串行通信 存储器 我们上的是嵌入式与接口技术这门课&#xff0c;存储器章节重点在于理解8086CPU的存储体结构&am…...

第 6 篇:衡量预测好坏 - 评估指标

第 6 篇&#xff1a;衡量预测好坏 - 评估指标 上一篇&#xff0c;我们小试牛刀&#xff0c;用朴素预测、平均法、移动平均法和季节性朴素预测这几种简单方法对未来进行了预测。我们还通过可视化将预测结果与真实值进行了对比。 但光靠眼睛看图来判断“哪个预测更好”往往是不…...

极刻AI搜v1.0 问一次问题 AI工具一起答

软件名&#xff1a;极刻AI搜 版本&#xff1a;v1.0 功能&#xff1a;囊括了互联网上比较好用的一些支持”搜索“的网站或者工具 开发平台&#xff1a;nodepythonweb 分类有&#xff1a; AI搜索&#xff08;支持智能问答的AI搜索引擎&#xff09; 常规搜索&#xff1a;&#xff…...

单片机 + 图像处理芯片 + TFT彩屏 进度条控件

进度条控件使用说明 概述 本进度条控件基于单片机 RA8889/RA6809 TFT开发&#xff0c;提供了简单易用的进度显示功能。控件支持多个进度条同时显示、自定义颜色、边框和标签等特性&#xff0c;适用于需要直观显示进度信息的各类应用场景。 特性 支持多个进度条同时显示可…...

RHCSA Linux系统 用户和组的管理

用户管理&#xff1a;增useradd 删userdel 改usermod 查id 组的管理&#xff1a;增groupadd 删groupdel 改groupmod 查groups /etc/default/useradd 即定义useradd默认参数&#xff0c;也定义了/etc/passwd &#xff0c;/etc/shadow&#xff0c;/etc/group&#…...

2025年pta团队设计天梯赛题解

题解不全&#xff0c;望见谅 L1-1 珍惜生命 题目 前辈工程师 Martin Golding 教育我们说&#xff1a;“Always code as if the guy who ends up maintaining your code will be a violent psychopath who knows where you live.”&#xff08;写代码的时候&#xff0c;总是要…...

Vue---vue2和vue3的生命周期

核心生命周期对比 生命周期阶段Vue 2 钩子Vue 3 Composition API​​初始化​​beforeCreate无&#xff08;使用 setup() 替代&#xff09;​​初始化完成​​created无&#xff08;使用 setup() 替代&#xff09;​​挂载前​​beforeMountonBeforeMount​​挂载完成​​moun…...

C#常用LINQ

在开发时发现别人的代码使用到了LINQ十分便捷且清晰&#xff0c;这里记录一下常用LINQ和对应的使用。参考链接&#xff1a;LINQ 菜鸟教程 使用的学生类和字符串用于测试 public class Student {public int StudentID;public string StudentName;public int Age; }Student[] st…...

Java--数组的应用

一、数组的地址值 数组的地址值表示数组在内存中的位置。 [I1eb44e46 [ &#xff1a;表示当前是一个数组I&#xff1a;表示当前数组是int类型&#xff1a;表示一个间隔符号&#xff08;固定格式&#xff09;1eb44e46&#xff1a;数组真正的地址值&#xff08;十六进制&#…...

PostgreSQL基础

一、PostgreSQL介绍 PostgreSQL是一个功能强大的 开源 的关系型数据库。底层基于C实现。 PostgreSQL的开源协议和Linux内核版本的开源协议是一样的。。BDS协议&#xff0c;这个协议基本和MIT开源协议一样&#xff0c;说人话&#xff0c;就是你可以对PostgreSQL进行一些封装&a…...

Linux系统管理与编程13:基于CentOS7.x的LAMP环境部署

兰生幽谷&#xff0c;不为莫服而不芳&#xff1b; 君子行义&#xff0c;不为莫知而止休。 一、实验目标 1.理解Apache服务器原理 2.掌握Apache服务器的配置文件 3.具备安装Mysql数据库能力 4.具备安装Apache服务器能力 5.具备PHP与数据库连接能力 6.具备Apache、Mysql、…...

浅谈AI致幻

文章目录 当前形势下存在的AI幻觉&#xff08;AI致幻&#xff09;什么是AI幻觉AI幻觉的类型为什么AI会产生幻觉AI幻觉的危害与影响当前应对AI幻觉的技术与方法行业与学术界的最新进展未来挑战与展望结论 当前形势下存在的AI幻觉&#xff08;AI致幻&#xff09; 什么是AI幻觉 …...

【架构】-- StarRocks 和 Doris 介绍与选型建议

StarRocks 和 Doris 的介绍 随着大数据分析需求的不断增长,企业对高性能、低延迟的分析型数据库提出了更高的要求。StarRocks 和 Apache Doris 是当前主流的开源 MPP(Massively Parallel Processing)数据库系统,广泛应用于实时分析、报表生成和数据仓库等场景。本文将从架…...

【SF顺丰】顺丰开放平台API对接(注册、API测试篇)

1.注册开发者账号 注册地址&#xff1a;顺丰企业账户中心 2.登录开发平台 登录地址&#xff1a;顺丰开放平台 3.开发者对接 点击开发者对接 4.创建开发对接应用 开发者应用中“新建应用”创建应用&#xff0c;最多创建应用限制数量5个 注意&#xff1a;需要先复制保存生产校验…...

C语言高频面试题——常量指针与指针常量区别

1. 常量指针&#xff08;Pointer to Constant&#xff09; 定义&#xff1a; 常量指针是指向一个常量数据的指针&#xff0c;即指针指向的内容不能通过该指针被修改。 语法&#xff1a; const int* ptr;或者&#xff1a; int const* ptr;解释&#xff1a; const修饰的是指…...

Novartis诺华制药社招入职综合能力测评真题SHL题库考什么?

一、综合能力测试 诺华制药的入职测评中&#xff0c;综合能力测试是重要的一部分&#xff0c;主要考察应聘者的问题解决能力、数值计算能力和逻辑推理能力。测试总时长为46分钟&#xff0c;实际作答时间为36分钟&#xff0c;共24题。题型丰富多样&#xff0c;包括图形变换题、分…...

网页下载的m3u8格式文件使用FFmpeg转为MP4

FFmpeg 是一个强大的开源音视频处理工具&#xff0c;可以直接将 M3U8 合并并转换为 MP4。 1.步骤&#xff1a; 下载 FFmpeg 官网&#xff1a;https://ffmpeg.org/ Windows 用户可以直接下载 静态构建版本&#xff08;Static Build&#xff09;&#xff0c;解压后即可使用。 2…...

Java 并发包核心机制深度解析:锁的公平性、异步调度、AQS 原理全解

&#x1f9e0; Java 并发包核心机制深度解析&#xff1a;锁的公平性、异步调度、AQS 原理全解 Java 并发编程的地基是 java.util.concurrent&#xff0c;但真正驱动这个系统的&#xff0c;是它背后隐藏的三根支柱&#xff1a; ReentrantLock 的公平/非公平调度策略Completabl…...

μC/OS 版本演进过程 | uC/OS-II 和 uC/OS-III 有什么区别?

uC/OS 系列是由 Jean J. Labrosse 开发的一套嵌入式实时操作系统&#xff08;RTOS&#xff09;&#xff0c;以其高质量源码和清晰的结构&#xff0c;在嵌入式教学和某些工业项目中有着广泛影响。该系统主要包含两个版本&#xff1a;uC/OS-II 和 uC/OS-III。 本文将带你了解这两…...