实现光线追踪需从摄像机向像素发射光线,计算与球体交点并着色。1. 定义Vec3和Ray类用于数学运算;2. 通过解二次方程实现光线与球体求交;3. 使用Lambert模型根据法线与光照方向夹角计算漫反射颜色;4. 在主循环中遍历像素生成光线,检测交点后着色并写入图像;5. 最终以PPM格式输出暖色调球体渲染结果。
实现一个简单的光线追踪渲染器是理解计算机图形学核心概念的绝佳方式。用C++从零开始写一个基础的光线追踪器,不需要复杂的库或框架,只需要基本的数学知识和对光线与物体交互的理解。
光线追踪的核心思想是从摄像机出发,向场景中的每个像素发射一条光线,然后计算这条光线是否与场景中的物体相交,如果相交,就根据光照模型计算该点的颜色。
主要步骤包括:
首先需要几个关键类:三维向量、光线、物体(如球体)、材质和颜色输出。
// Vector3.h
struct Vec3 {
float x, y, z;
Vec3(float x = 0, float y = 0, float z = 0) : x(x), y(y), z(z) {}
Vec3 operator+(const Vec3& b) const { return Vec3(x + b.x, y + b.y, z + b.z); }
Vec3 operator-(const Vec3& b) const { return Vec3(x - b.x, y - b.y, z - b.z); }
Vec3 operator*(float t) const { return Vec3(x * t, y * t, z * t); }
float dot(const Vec3& b) const { return x * b.x + y * b.y + z * b.z; }
Vec3 cross(const Vec3& b) const { return Vec3(y * b.z - z * b.y, z * b.x - x * b.z, x * b.y - y * b.x); }
Vec3 normalize() const { float len = sqrtf(dot(*this)); return len > 0 ? (*this) * (1.0f / len) : *this; }
};
// Ray.h
struct Ray {
Vec3 origin, direction;
Ray(const Vec3& o, const Vec3& d) : origin(o), direction(d.normalize()) {}
Vec3 pointAt(float t) const { return origin + direction * t; }
};
球体是最容易求交的几何体之一。给定球心和半径,利用几何公式解二次方程判断是否有交点。
bool intersectSphere(const Ray& ray, const Vec3& center, float radius, float& t) {
Vec3 oc = ray.origin - center
;
float a = ray.direction.dot(ray.direction);
float b = 2.0f * oc.dot(ray.direction);
float c = oc.dot(oc) - radius * radius;
float discriminant = b * b - 4 * a * c;
if (discriminant
t = (-b - sqrtf(discriminant)) / (2.0f * a);
return t > 0.001f; // 避免自相交
}
使用 Lambert 漫反射模型进行着色。假设有一个方向光,颜色由法向与光照方向的夹角决定。
Vec3 shade(const Ray& ray, const Vec3& hitPoint, const Vec3& normal, const Vec3& lightDir) {
float diff = fmaxf(0.0f, normal.dot(lightDir.normalize() * -1));
return Vec3(1.0f, 0.8f, 0.6f) * diff; // 暖色调漫反射
}
设置图像分辨率,遍历每个像素生成光线,尝试与球体相交,并记录颜色。
int main() {
const int width = 800, height = 600;
unsigned char* image = new unsigned char[width * height * 3];
Vec3 camera(0, 0, -5);
Vec3 sphereCenter(0, 0, 0);
float sphereRadius = 1.0f;
Vec3 lightDir(-1, -1, -1);
for (int y = 0; y < height; y++) {
for (int x = 0; x < width; x++) {
float u = (2.0f * (x + 0.5f) / width - 1.0f) * (float)width / height;
float v = (2.0f * (y + 0.5f) / height - 1.0f);
Ray ray(camera, Vec3(u, v, 0) - camera);
float t;
if (intersectSphere(ray, sphereCenter, sphereRadius, t)) {
Vec3 hit = ray.pointAt(t);
Vec3 normal = (hit - sphereCenter).normalize();
Vec3 color = shade(ray, hit, normal, lightDir);
int idx = (y * width + x) * 3;
image[idx + 0] = (unsigned char)(color.x * 255);
image[idx + 1] = (unsigned char)(color.y * 255);
image[idx + 2] = (unsigned char)(color.z * 255);
} else {
int idx = (y * width + x) * 3;
image[idx + 0] = image[idx + 1] = image[idx + 2] = 128; // 背景色
}
}
}
// 保存为PPM格式(简单图像格式)
FILE* f = fopen("render.ppm", "w");
fprintf(f, "P3\n%d %d\n255\n", width, height);
for (int i = 0; i < width * height * 3; i += 3) {
fprintf(f, "%d %d %d ", image[i], image[i+1], image[i+2]);
}
fclose(f);
delete[] image;
return 0;
}
基本上就这些。这个渲染器虽然只能画一个球,但已经包含了光线追踪的核心流程:光线生成、求交、着色、输出。后续可以扩展支持多个物体、镜面反射、阴影、纹理、BVH加速等。
# c++
# 计算机
# ai
# Float
# if
# const
# bool
# char
# int
# 循环
# 数据结构
# operator
# len
# this
# 遍历
# 球心
# 几个
# 景中
# 渲染器
# 多个
# 不需要
# 是从
# 这条
# 只需要
相关文章:
网站制作软件免费下载安装,有哪些免费下载的软件网站?
建站之星后台管理如何实现高效配置?
,石家庄四十八中学官网?
教育培训网站制作流程,请问edu教育网站的域名怎么申请?
宝塔新建站点为何无法访问?如何排查?
个人网站制作流程图片大全,个人网站如何注销?
北京网站制作公司哪家好一点,北京租房网站有哪些?
如何确保FTP站点访问权限与数据传输安全?
制作证书网站有哪些,全国城建培训中心证书查询官网?
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
网站制作和推广的区别,想自己建立一个网站做推广,有什么快捷方法马上做好一个网站?
如何基于云服务器快速搭建个人网站?
C++中引用和指针有什么区别?(代码说明)
高防服务器如何保障网站安全无虞?
已有域名和空间如何搭建网站?
如何用西部建站助手快速创建专业网站?
车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?
做企业网站制作流程,企业网站制作基本流程有哪些?
建站之星安装提示数据库无法连接如何解决?
三星网站视频制作教程下载,三星w23网页如何全屏?
如何在云虚拟主机上快速搭建个人网站?
定制建站哪家更专业可靠?推荐榜单揭晓
网站制作壁纸教程视频,电脑壁纸网站?
广平建站公司哪家专业可靠?如何选择?
香港代理服务器配置指南:高匿IP选择、跨境加速与SEO优化技巧
如何在腾讯云服务器快速搭建个人网站?
如何在橙子建站上传落地页?操作指南详解
*服务器网站为何频现安全漏洞?
百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?
PHP正则匹配日期和时间(时间戳转换)的实例代码
详解jQuery停止动画——stop()方法的使用
如何在自有机房高效搭建专业网站?
制作ppt免费网站有哪些,有哪些比较好的ppt模板下载网站?
建站主机选购指南:核心配置与性价比推荐解析
网站制作报价单模板图片,小松挖机官方网站报价?
如何在Windows环境下新建FTP站点并设置权限?
php8.4新语法match怎么用_php8.4match表达式替代switch【方法】
建站之星伪静态规则如何正确配置?
如何在香港免费服务器上快速搭建网站?
制作网站的公司有哪些,做一个公司网站要多少钱?
如何配置FTP站点权限与安全设置?
网站图片在线制作软件,怎么在图片上做链接?
如何制作新型网站程序文件,新型止水鱼鳞网要拆除吗?
建站之星CMS五站合一模板配置与SEO优化指南
头像制作网站在线制作软件,dw网页背景图像怎么设置?
如何选择网络建站服务器?高效建站必看指南
建站主机空间推荐 高性价比配置与快速部署方案解析
建站之星如何快速更换网站模板?
洛阳网站制作公司有哪些,洛阳的招聘网站都有哪些?
如何选择域名并搭建高效网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。