从零开始使用C ++编写Ray-Tracer
admin
2023-02-11 05:40:07
0

光线追踪是一种基于光传播模拟的3D渲染技术。该技术能够产生非常高的视觉真实感。

原理很简单。它包括滑动每个像素的屏幕像素。对于每个像素,请查看从相机经过该像素的光线是否会与场景中的任何对象相交。像素采用对象的颜色。如果不相交,则像素采用背景色。实际上,像素并不会照原样呈现对象颜色,而是通过称为着色的过程计算出的颜色。着色的目的是使渲染更加真实感。
摄像机的位置以及场景中的所有物体均由3d向量表示,颜色也由3d向量表示。因此,我们需要一直操纵向量。

#ifndef Vect_h
#define Vect_h

#include 

using namespace std;

class Vec3 {
public:
    float X;
    float Y;
    float Z;
    Vec3(float x, float y, float z) : X(x), Y(y), Z(z) {}
    Vec3 operator +(Vec3 vec) {
        return Vec3(X+vec.X,Y+vec.Y,Z+vec.Z);
    }
    Vec3 operator -(Vec3 vec) {
        return Vec3(X-vec.X,Y-vec.Y,Z-vec.Z);
    }
    Vec3 operator *(float scalar) {
        return Vec3(scalar*X,scalar*Y,scalar*Z);
    }
    float dot(Vec3 vec) {
        return X*vec.X+Y*vec.Y+Z*vec.Z;
    }
    float norm() {
        return sqrt(dot(*this));
    }
    Vec3 normalize() {
        return (*this)*(1/norm());
    }

};

#endif

射线与球体的交点:
从零开始使用C ++编写Ray-Tracer
以c为中心,半径为r的球面的方程为:
从零开始使用C ++编写Ray-Tracer
射线(一条线)的方程为:
从零开始使用C ++编写Ray-Tracer
其中o代表射线的原点,d代表射线的方向,t是参数。
那么对于相交点,t必须满足:
从零开始使用C ++编写Ray-Tracer
注意:
从零开始使用C ++编写Ray-Tracer
然后 :
从零开始使用C ++编写Ray-Tracer
这是二次方程。通过添加单位向量d,我们可以:
从零开始使用C ++编写Ray-Tracer
解(如果有)是:
从零开始使用C ++编写Ray-Tracer

    bool Sphere::intersect(Ray& ray, float &t) {
        Vec3 o = ray.get_origin();
        Vec3 d = ray.get_direction();

        Vec3 v = o - Center;

        const float b = 2 * v.dot(d);
        const float c = v.dot(v) - Radius*Radius;
        float delta = b*b - 4 * c;

        if (delta < 1e-4)
            return false;

        const float t1 = (-b - sqrt(delta))/2;
        const float t2 = (-b + sqrt(delta))/2;

        t = (t1 < t2) ? t1 : t2; // get the first intersection only

        return true;
    }

着色:
在开始之前,让我们介绍一些符号。
从零开始使用C ++编写Ray-Tracer
从要着色点到摄像机的单位向量V。
在所考虑点垂直于球体的法向量N。
易得给定球面一点的法向量:


    Vec3 Sphere::get_normal(Vec3 p) {
        return (p - Center) * (-1/(Radius));
    }

因为:在考虑的点p上的每个法向矢量必须与将p连接到球体中心的半径共线。
简单着色工作很好(正面比,此着色器将返回着色法线和入射光线方向的点积的绝对值,在其他渲染器中,也称为入射。):
facing ratio = V . N
它是V和N之间的夹角的余弦。我们可以使用它来对球体进行如下着色:

Vec3 Shading(Ray ray, Sphere sphere) { 
  Vec3 color(0, 0, 0);
  float t;
  if (sphere.intersect(ray, t)) {
    Vec3 V = ray.get_direction();
    Vec3 P = ray.get_origin() +  V * t;
    Vec3 N = sphere.get_normal(P);

    float facing_ratio = N.dot(V);

    color = sphere.get_color() * (facing_ratio * 0.5);
  }
  return color;
}

结果如下:
从零开始使用C ++编写Ray-Tracer

相关内容

热门资讯

【第一资讯】“新老夫子牛牛到底... 网上科普关于“新老夫子牛牛有没有挂”话题很是火热,小编也是针对新老夫子牛牛作*弊开挂的方法以及开挂对...
今日重大通报“来几局到底有挂吗... 家人们!今天小编来为大家解答来几局透视挂怎么安装这个问题咨询软件客服徽9784099的挂在哪里买很多...
原创 生... 来源:网络消息 2025年12月15日,生命科学开放联盟2025年度交流活动在广州举办,南京大学、...
字节三战春晚!豆包能否再造“微... 来源:时代周报-时代在线 图源:视觉中国 近来,豆包动作密集且重磅。 有报道称,火山引擎将成为20...
今日重大消息“728究竟有挂吗... 今日重大消息“728究竟有挂吗?”(太坑了果然有挂)您好,728这个游戏其实有挂的,确实是有挂的,需...
【第一资讯】“齐聚天下怎么开挂... 网上科普关于“齐聚天下有没有挂”话题很是火热,小编也是针对齐聚天下作*弊开挂的方法以及开挂对应的知识...
今日重磅消息“非凡贪玩开挂神器... 网上科普关于“非凡贪玩有没有挂”话题很是火热,小编也是针对非凡贪玩作*弊开挂的方法以及开挂对应的知识...
今日重大通报“决胜麻将是不是有... 网上科普关于“决胜麻将有没有挂”话题很是火热,小编也是针对决胜麻将作*弊开挂的方法以及开挂对应的知识...
最新引进“熊猫麻将到底是不是挂... 家人们!今天小编来为大家解答熊猫麻将透视挂怎么安装这个问题咨询软件客服徽4282891的挂在哪里买很...
我来教教您“闽游麻将究竟有挂吗... 家人们!今天小编来为大家解答闽游麻将透视挂怎么安装这个问题咨询软件客服徽9752949的挂在哪里买很...