答案:文章介绍了在C++图形学编程中实现基础数学库的方法,包含Vec3类用于三维向量运算如加减、点积、叉积和单位化,Mat4类实现4x4矩阵的乘法与向量变换,支持平移、旋转、缩放及透视投影等操作,并通过Transform工具类提供常用变换函数,最后给出组合变换的使用示例,适用于教学和原型开发。
在C++图形学编程中,向量(Vec3)和4x4矩阵(Mat4)是基础中的基础。实现一个简单的数学库能帮助你理解底层运算,同时为后续的渲染、变换、光照计算打下基础。下面教你如何从零开始写一个轻量级但实用的 Vec3 和 Mat4 类。
Vec3 表示三维空间中的点或方向,支持加减、数乘、点积、叉积等操作。
基本结构:
// Vec3.h #pragma once #includeclass Vec3 { public: float x, y, z;
// 构造函数
Vec3() : x(0), y(0), z(0) {}
Vec3(float x, float y, float z) : x(x), y(y), z(z) {}
// 向量加法
Vec3 operator+(const Vec3& other) const {
return Vec3(x + other.x, y + other.y, z + other.z);
}
// 向量减法
Vec3 operator-(const Vec3& other) const {
return Vec3(x - other.x, y - other.y, z - other.z);
}
// 数乘
Vec3 operator*(float scalar) const {
return Vec3(x * scalar, y * scalar, z * scalar);
}
// 点积
float dot(const Vec3& other) const {
return x * other.x + y * other.y + z * other.z;
}
// 叉积
Vec3 cross(const Vec3& other) const {
return Vec3(
y * other.z - z * other.y,
z * other.x - x * other.z,
x * other.y - y * other.x
);
}
// 向量长度
float length() const {
return std::sqrt(x*x + y*y + z*z);
}
// 单位化
Vec3 normalize() const {
float len = length();
return len != 0 ? *this * (1.0f / len) : Vec3(0, 0, 0);
}};
// 支持 scalar vector inline Vec3 operator(float scalar, const Vec3& v) { return v * scalar; }
Mat4 用于表示空间变换:平移、旋转、缩放和投影。我们用列主序存储16个浮点数。
// Mat4.h #pragma onceclass Mat4 { public: float m[16]; // 列主序:m[0-3] 是第一列
// 单位矩阵构造
Mat4() {
for (int i = 0; i zuojiankuohaophpcn 16; i++) {
m[i] = (i % 5 == 0) ? 1.0f : 0.0f; // 对角线为1
}
}
// 手动设置元素
void set(int row, int col, float value) {
m[col * 4 + row] = value; // 列主序索引
}
// 矩阵乘法
Mat4 operator*(const Mat4& other) const {
Mat4 result;
for (int i = 0; i zuojiankuohaophpcn 4; i++) {
for (int j = 0; j zuojiankuohaophpcn 4; j++) {
float sum = 0;
for (int k = 0; k zuojiankuohaophpcn 4; k++) {
sum += m[i * 4 + k] * other.m[k * 4 + j];
}
result.m[i * 4 + j] = sum;
}
}
return result;
}
// 矩阵与向量相乘(齐次坐标)
Vec3 operator*(const Vec3& v) const {
float x = m[0]*v.x + m[4]*v.y + m[8]*v.z + m[12];
float y = m[1]*v.x + m[5]*v.y + m[9]*v.z + m[13];
float z = m[2]*v.x + m[6]*v.y + m[10]*v.z + m[14];
float w = m[3]*v.x + m[7]*v.y + m[11]*v.z + m[15];
if (w != 0 && w != 1) {
return Vec3(x/w, y/w, z/w);
}
return Vec3(x, y, z);
}};
静态函数生成特定变换矩阵,方便调用。
// TransformUtils.h #pragma once #include "Mat4.h" #includeclass Transform { public: // 平移矩阵 static Mat4 translate(float x, float y, float z) { Mat4 mat; mat.set(0, 3, x); mat.set(1, 3, y); mat.set(2, 3, z); return mat; }
// 缩放矩阵
static Mat4 scale(float x, float y, float z) {
Mat4 mat;
mat.set(0, 0, x);
mat.set(1, 1, y);
mat.set(2, 2, z);
return mat;
}
// 绕X轴旋转
static Mat4 rotateX(float rad) {
Mat4 mat;
float c = cosf(rad);
float s = sinf(rad);
mat.set(1, 1, c); mat.set(1, 2, -s);
mat.set(2, 1, s); mat.set(2, 2, c);
return mat;
}
// 绕Y轴旋转
static Mat4 rotateY(float rad) {
Mat4 mat;
float c = cosf(rad);
float s = sinf(rad);
mat.set(0, 0, c); mat.set(0, 2, s);
mat.set(2, 0, -s); mat.set(2, 2, c);
return mat;
}
// 绕Z轴旋转
static Mat4 rotateZ(float rad) {
Mat4 mat;
float c = cosf(rad);
float s = sinf(rad);
mat.set(0, 0, c); mat.set(0, 1, -s);
mat.set(1, 0, s); mat.set(1, 1, c);
return mat;
}
// 透视投影矩阵(简化版)
static Mat4 perspective(float fov, float aspect, float near, float far) {
Mat4 mat;
float f = 1.0f / tanf(fov * 0.5f);
mat.set(0, 0, f / aspect);
mat.set(1, 1, f);
mat.set(2, 2, (far + near) / (near - far));
mat.set(2, 3, (2 * far * near) / (near - far));
mat.set(3, 2, -1);
mat.set(3, 3, 0);
return mat;
}};
int main() { Vec3 pos(1, 0, 0);
Mat4 model = Transform::translate(2, 0, 0);
Mat4 rot = Transform::rotateZ(3.14159f / 4.0f);
Mat4 scale = Transform::scale(2, 2, 2);
Mat4 transform = model * rot * scale;
Vec3 transformed = transform * pos;
std::cout zuojiankuohaophpcnzuojiankuohaophpcn "Transformed: ("
zuojiankuohaophpcnzuojiankuohaophpcn transformed.x zuojiankuohaophpcnzuojiankuohaophpcn ", "
zuojiankuohaophpcnzuojiankuohaophpcn transformed.y zuojiankuohaophpcnzuojiankuohaophpcn ", "
zuojiankuohaophpcnzuojiankuohaophpcn transformed.z zuojiankuohaophpcnzuojiankuohaophpcn ")\n";
return 0;}
基本上就这些。这个简易数学库没有过度优化,但足够教学和原型开发使用。你可以逐步添加更多功能,比如欧拉角转矩阵、LookAt 矩阵、逆矩阵等。关键是理解每一步运算的几何意义。
# 工具
# ai
# c++
# ios
# stream
# cos
# Static
# Float
# include
# const
# int
# class
# public
# operator
# transform
# 加减
# 你可以
# 适用于
# 教你如何
# 从零开始
# 简化版
# 欧拉
# 浮点数
# 帮助你
# len
相关文章:
建站之星如何通过成品分离优化网站效率?
大同网页,大同瑞慈医院官网?
建站之星代理如何优化在线客服效率?
c# await 一个已经完成的Task会发生什么
如何快速配置高效服务器建站软件?
C++中的Pimpl idiom是什么,有什么好处?(隐藏实现)
建站主机类型有哪些?如何正确选型
车管所网站制作流程,交警当场开简易程序处罚决定书,在交警网站查询不到怎么办?
网站制作和推广的区别,想自己建立一个网站做推广,有什么快捷方法马上做好一个网站?
如何优化Golang Web性能_Golang HTTP服务器性能提升方法
javascript中对象的定义、使用以及对象和原型链操作小结
建站主机选虚拟主机还是云服务器更好?
内部网站制作流程,如何建立公司内部网站?
潮流网站制作头像软件下载,适合母子的网名有哪些?
广州网站建站公司选择指南:建站流程与SEO优化关键词解析
太平洋网站制作公司,网络用语太平洋是什么意思?
建站中国官网:模板定制+SEO优化+建站流程一站式指南
如何在景安服务器上快速搭建个人网站?
建站主机选哪种环境更利于SEO优化?
如何在建站之星网店版论坛获取技术支持?
新网站制作渠道有哪些,跪求一个无线渠道比较强的小说网站,我要发表小说?
如何在Golang中实现微服务服务拆分_Golang微服务拆分与接口管理方法
上海网站制作网页,上海本地的生活网站有哪些?最好包括生活的各个方面的?
代购小票制作网站有哪些,购物小票的简要说明?
深圳网站制作的公司有哪些,dido官方网站?
建站主机功能解析:服务器选择与快速搭建指南
独立制作一个网站多少钱,建立网站需要花多少钱?
长沙做网站要多少钱,长沙国安网络怎么样?
婚礼视频制作网站,学习*后期制作的网站有哪些?
如何通过二级域名建站提升品牌影响力?
如何快速生成凡客建站的专业级图册?
如何用狗爹虚拟主机快速搭建网站?
建站之星如何防范黑客攻击与数据泄露?
建站之星如何优化SEO以实现高效排名?
制作网站的软件免费下载,免费制作app哪个平台好?
红河网站制作公司,红河事业单位身份证如何上传?
外贸公司网站制作哪家好,maersk船公司官网?
建站之星2.7模板:企业网站建设与h5定制设计专题
网站制作软件有哪些,制图软件有哪些?
高防服务器租用指南:配置选择与快速部署攻略
如何打造高效商业网站?建站目的决定转化率
如何在IIS中新建站点并配置端口与物理路径?
如何在Golang中引入测试模块_Golang测试包导入与使用实践
,购物网站怎么盈利呢?
如何获取上海专业网站定制建站电话?
儿童网站界面设计图片,中国少年儿童教育网站-怎么去注册?
ppt在线制作免费网站推荐,有什么下载免费的ppt模板网站?
香港服务器网站搭建教程-电商部署、配置优化与安全稳定指南
如何选择网络建站服务器?高效建站必看指南
西安专业网站制作公司有哪些,陕西省建行官方网站?
*请认真填写需求信息,我们会在24小时内与您取得联系。