IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> C++知识库 -> 模型矩阵分解 -> 正文阅读

[C++知识库]模型矩阵分解

文章目录

1. 正文

通常来说,模型矩阵(R)的一种比较好的级联方式为:先缩放(S),再旋转?,最后平移(T):
R = T ? R ? S \textbf{R} = \textbf{T} * \textbf{R} * \textbf{S} R=T?R?S

如果不考虑缩放变换,那么模型变换实际上是一种刚体变换。此时四维模型矩阵的左上角3X3矩阵就是旋转矩阵,第四列就是平移量。但是加上缩放变换,就变成一个复杂的问题了。可以参考GLM的实现:

#include <iostream>
#include <glm/gtx/matrix_decompose.hpp>
#include <glm/gtx/euler_angles.hpp>
#include <glm/gtx/quaternion.hpp>

static void PrintMat(const glm::mat4& m)
{
	for (int i = 0; i < 4; i++)
	{
		for (int j = 0; j < 4; j++)
		{
			printf("%.9lf\t", m[i][j]);
		}
		printf("\n");
	}
}

static void PrintVec3(const glm::vec3& v)
{
	printf("%lf\t%lf\t%lf\n", v.x, v.y, v.z);
}

static void PrintVec4(const glm::vec4& v)
{
	printf("%lf\t%lf\t%lf\t%lf\n", v.x, v.y, v.z, v.w);
}


int main()
{
	//平移矩阵
	glm::vec3 position(100,200,300);
	glm::mat4 translationMatrix = glm::translate(glm::identity<glm::mat4>(),
		position);

	//旋转矩阵
	glm::vec3 eulerAngles(45,60,70);
	glm::mat4 rotationMatrix = glm::eulerAngleYXZ(glm::radians(eulerAngles.y),
		glm::radians(eulerAngles.x), glm::radians(eulerAngles.z));

	//缩放矩阵
	glm::vec3 scalePre(2, 3, 4);
	glm::mat4 scaleMatrix = glm::scale(glm::identity<glm::mat4>(), scalePre);

	glm::mat4 modelMatrix = translationMatrix * rotationMatrix * scaleMatrix;
	glm::vec3 scale;
	glm::quat quaternion;
	glm::vec3 translation;
	glm::vec3 skew;
	glm::vec4 perspective;
	
	glm::decompose(modelMatrix, scale, quaternion, translation, skew, perspective);
		
	PrintVec3(translation);
	PrintVec3(scale);
	PrintVec3(skew);
	PrintVec4(perspective);
	
	glm::mat4 rotationMatrix1 = glm::toMat4(quaternion);
	glm::vec3 euler(0, 0, 0);
	glm::extractEulerAngleYXZ(rotationMatrix1, euler.y, euler.x, euler.z);
	euler.y = glm::degrees(euler.y);
	euler.x = glm::degrees(euler.x);
	euler.z = glm::degrees(euler.z);
	PrintVec3(euler);
}

运行结果如下:
imglink1

可以看出分解出来的缩放、旋转、平移和级联前的一致。

除了缩放、旋转和平移,GLM提供的模型矩阵分解的函数接口glm::decompose()还提供一个skew参数和perspective参数,暂时没弄明白其具体含义,留待以后研究。

2. 参考

  1. glm - Decompose mat4 into translation and rotation?
  2. GLM_GTX_matrix_decompose
  C++知识库 最新文章
【C++】友元、嵌套类、异常、RTTI、类型转换
通讯录的思路与实现(C语言)
C++PrimerPlus 第七章 函数-C++的编程模块(
Problem C: 算法9-9~9-12:平衡二叉树的基本
MSVC C++ UTF-8编程
C++进阶 多态原理
简单string类c++实现
我的年度总结
【C语言】以深厚地基筑伟岸高楼-基础篇(六
c语言常见错误合集
上一篇文章      下一篇文章      查看所有文章
加:2022-04-06 15:59:31  更:2022-04-06 16:03:22 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/10 20:59:25-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码