dx12 龙书第三章学习笔记 -- 变换

发布于:2023-01-25 ⋅ 阅读:(420) ⋅ 点赞:(0)

1.线性变换

\tau (v)=\tau (x,y,z)=\tau (x',y',z')  函数的输入和输出都是3D向量,我们称\tau为线性变换 

矩阵表示法:u = (x,y,z) = (xi,yj,zk)=x(1,0,0)+y(0,1,0)+z(0,0,1)

\\ \tau (u) =\tau (x,y,z)= \tau (xi+yj+zk)=x\tau (i)+y\tau(j)+z\tau(k)=uA=\begin{bmatrix} x & y & z \end{bmatrix} \begin{bmatrix} \gets \tau (i) \to \\ \gets \tau (j) \to \\ \gets \tau (k) \to \end{bmatrix}

所以已知一个线性变换,只要将i,j,z也就是标准基向量代入线性变换,就能构造一个变换矩阵

A:线性变换的矩阵表示法 -- 线性变换才能满足这种提取式

⭐若\tau为线性变换,当且仅当此函数具有下列性质:

\\\tau(u+v)=\tau(u)+\tau(v)\\ \tau(ku)=k\tau(u)

2.缩放:scaling

缩放变换是相对于当前坐标系中的原点,令向量在x、y、z轴上分别以系数s_x,s_y,s_z进行缩放

S=\begin{bmatrix} S_x & 0 & 0 \\ 0 & S_y & 0 \\ 0 & 0 & s_z \end{bmatrix} -- Sx Sy Sz就是xyz轴的缩放系数

[x, y, z]\begin{bmatrix} S_x & 0 & 0 \\ 0 & S_y & 0 \\ 0 & 0 & S_z \end{bmatrix}=[x',y',z']

左手坐标系的变换矩阵与坐标表示与右手有很大不同,变换矩阵乘在坐标右侧, 坐标以行向量形式存在

3.旋转:rotating

旋转矩阵证明(罗德里格旋转公式):假设这种情况,令向量v绕轴n以角θ进行旋转,在沿n轴从上至下俯瞰时,我们按顺时针方向来测量角θ,并假设||n||=1,v和n之间的夹角为α

旋转时我们考虑将向量v分解成平行于n轴的分量v_{//}(v_{proj})垂直于n轴的分量v_{\bot}(v_{prep}),平行于n轴的分量在旋转时是保持不变的

最终,旋转向量:

R_{n}(v)=proj_n(v)+R_{n}(v_\bot)

proj_n(v)=vcos\alpha =(n\cdot v)n   :v·n是大小,n是平行方向

R_{n}(v_\bot)=v_{\bot}\cdot (cos\theta+sin\theta)=cos\theta v_{\bot}+sin\theta (n\times v)    :在垂直于z轴的平面内旋转θ角度

确定大小:

n\cdot v = ||v||cos\alpha

||n\times v||=||v||sin\alpha =||v_\bot||    

确定方向:

n\times v  的方向就是旋转的"切方向"

因此推导出了旋转公式:

\\R_{n}(v)=proj_n(v)+R_{n}(v_\bot) \\=(v\cdot n)n +cos\theta v_{\bot}+sin\theta (n\times v)

=(v\cdot n)n +cos\theta (v-(n\cdot v)n)+sin\theta (n\times v) :因为v垂直=v-v平行

=cos\theta v + (1-cos\theta)(n\cdot v)n + sin\theta(n\times v)

要证明R_{n}(v)是线性变换,只要证明其满足上述两个性质⭐即可

cos\theta v + (1-cos\theta)(n\cdot v)n + sin\theta(n\times v)

①分配律:叉乘和点乘都满足分配律,所以第一条性质满足

②标量乘法提取:叉乘和点乘都满足能提取标量乘积,所以第二条性质满足

所以是线性变换 -- > 能写成矩阵表达式的方式

将各个标准基向量代入线性变换函数R_{n}(v),再把得到的向量分别作为矩阵的行向量,最终结果:

R_n=\begin{bmatrix} cos\theta+(1-cos\theta)x^2 & (1-cos\theta)xy+sin\theta z & (1-cos\theta)xz-sin\theta y \\ (1-cos\theta) xy-sin\theta z & cos\theta+(1-cos\theta)y^2 & (1-cos\theta)yz+sin\theta x\\ (1-cos\theta)xz+sin\theta y & (1-cos\theta)yz-sin\theta x & cos\theta+(1-cos\theta)z^2 \end{bmatrix}

其中:n=[x,y,z]

旋转矩阵有一个特殊的性质,每个行向量都为单位长度且两两正交,也就是说这些行向量都是规范正交的,因此旋转矩阵是正交矩阵  --- 正交矩阵的特性:逆矩阵等于转置矩阵

特别的:如果选择绕x轴、y轴或z轴旋转(n取(1,0,0)...)

R_x=\begin{bmatrix} 1 & 0 & 0 \\ 0 & cos\theta & sin\theta \\ 0 & -sin\theta & cos\theta \end{bmatrix} R_y=\begin{bmatrix} cos\theta & 0 & -sin\theta \\ 0 & 1 & 0 \\ sin\theta & 0 & cos\theta \\ \end{bmatrix} R_z=\begin{bmatrix} cos\theta & sin\theta & 0 \\ -sin\theta & cos\theta & 0 \\ 0 & 0 & 1 \\ \end{bmatrix}

注意:①这是在不考虑齐次坐标的情况下的旋转矩阵

②记忆方法:如若旋转x轴,则第一行第一列扣除,剩下的”余子阵“左上角为cosθ,右上角为sinθ以此类推

注意:旋转部分,是以原点为中心进行旋转! -- 如果想对一个对称图案进行旋转,请先确定其旋转中心 。

4.齐次坐标:homogeneous coordinate

将坐标扩充为四元组,第四个坐标的取值w根据坐标描述对象的不同而不同:

(x,y,z,0) 表示向量

(x,y,z,1) 表示点

好处:①向量不考虑平移操作,设置w=0可以防止收到平移操作的影响

②点-点=向量 点+向量=点 向量+向量=向量

5.仿射变换

仿射变换=线性变换+平移

\\ \alpha(u)=\tau(u)+b \\ \alpha(u)=uA+b=[x,y,z]\begin{bmatrix} A_{11} & A_{12} & A_{13}\\ A_{21} & A_{22} & A_{23}\\ A_{31} & A_{32} & A_{33} \end{bmatrix}+[b_x,b_y,b_z]=[x',y',z']

扩充为齐次坐标后:

[x,y,z,1]\begin{bmatrix} A_{11} & A_{12} & A_{13} & 0\\ A_{21} & A_{22} & A_{23}& 0\\ A_{31} & A_{32} & A_{33}& 0 \\ b_x & b_y & b_z & 1 \end{bmatrix}=[x',y',z',1]

①如果输入的w=0,那么0*b_x=0,所以没有执行平移操作

②输入w不管是多少,仿射变换矩阵最后一列都是[0,0,0,1]^T,得到的新坐标w仍然为原值,所以进行仿射变换后不会改变w的值

注意,不管是缩放还是平移等变换矩阵,如果再乘以变换矩阵的逆,那么物体就没有移动,所以从理论上我们就能联想到平移矩阵和缩放矩阵的逆是如何书写的

6.变换的组合:

变换矩阵的组合:SRT  --  S:缩放矩阵  R:旋转矩阵  T:平移矩阵

我们往往会先把各种变换矩阵相乘,在对每一个顶点操作,免得每个顶点都挨着乘上不同的变换矩阵,影响性能 -- 但是注意,矩阵乘法不满足交换律,所以矩阵乘法的顺序很重要

7.坐标变换:

①向量的坐标变换:

 对于框架A:\overrightarrow{p_A} =x\overrightarrow{u} +y\overrightarrow{v},对于框架B:\overrightarrow{p_B} =x\overrightarrow{u_B} +y\overrightarrow{v_B}=(x',y')

其中x,y是P点在框架A中的坐标,\overrightarrow{u_B},\overrightarrow{v_B}是框架B的xy坐标轴相对于框架A的坐标 

 已知P点在A框架中的坐标,以及A框架中坐标轴相对于B框架的坐标,即可求得P_B

现将向量推广到3D空间,如果P_A=(x,y,z),那么P_B=(x\overrightarrow{u_B},y \overrightarrow{v_B} ,z\overrightarrow{w_B})

②点的坐标变换:

点的坐标变换,相比于向量的坐标变换,要额外考虑坐标系的平移,坐标系的原点不在同一处

所以加上平移向量:\overrightarrow{p_B} =x\overrightarrow{u_B} +y\overrightarrow{v_B}+z\overrightarrow{w_B}+\overrightarrow{Q_B}    

\overrightarrow{Q_B}:框架A中原点在框架B中的位置坐标

③(总结)坐标变换的矩阵表示:

根据线性变换和齐次坐标(\overrightarrow{Q_B}对应w坐标)的概念,我们可以把\overrightarrow{p_B} =x\overrightarrow{u_B} +y\overrightarrow{v_B}+z\overrightarrow{w_B}+\overrightarrow{Q_B}拆成矩阵相乘形式:

[x',y',z',w]=[x,y,z,w]\begin{bmatrix} \gets u_B \to \\ \gets v_B \to \\ \gets w_B \to \\ \gets Q_B \to \end{bmatrix}=[x,y,z,w]\begin{bmatrix} u_x & u_y & u_z & 0\\ v_x & v_y & v_z & 0\\ w_x & w_y & w_z & 0\\ Q_x & Q_y & Q_z & 1 \end{bmatrix}

 坐标变换矩阵:\begin{bmatrix} u_x & u_y & u_z & 0\\ v_x & v_y & v_z & 0\\ w_x & w_y & w_z & 0\\ Q_x & Q_y & Q_z & 1 \end{bmatrix}   坐标转换矩阵的逆: \begin{bmatrix} u_x & v_x & w_x & 0\\ u_y & v_y & w_y & 0\\ u_z & v_z & w_z & 0\\ -Q_x & -Q_y & -Q_z & 1 \end{bmatrix}

-- 本质上是一个旋转加平移矩阵[旋转矩阵的逆就是其转置矩阵、平移矩阵的逆就是取负数] -- 能把坐标变换,理解成与几何变换相同的数学形式,可谓是殊途同归RT=\begin{bmatrix} u_x & u_y & u_z & 0\\ v_x & v_y & v_z & 0\\ w_x & w_y & w_z & 0\\ 0 & 0 & 0 & 1 \end{bmatrix}\cdot \begin{bmatrix} 1 & 0 & 0 & 0\\ 0 & 1 & 0 & 0\\ 0 & 0 & 1 & 0\\ Q_x & Q_y & Q_z & 1 \end{bmatrix}

在后续DX12管线处理中,我们需要得到观察矩阵(view matrix),观察矩阵可以从世界空间到观察空间的转换,我们可以通过求W的逆,将坐标从世界空间变换到观察空间,DirectXMath提供专门的函数XMMatrixLookAtLH()得到观察矩阵,也可以自行构造(龙书第15章书写摄像机类时自行构造了观察矩阵)

具体代码:

// 代码在龙书P499~500页
float x = -XMVectorGetX(XMVector3Dot(P, R));
float y = -XMVectorGetX(XMVector3Dot(P, U));
float z = -XMVectorGetX(XMVector3Dot(P, L));

// mRight:摄像机右侧 -- 观察坐标系+x轴
mView(0, 0) = mRight.x;
mView(1, 0) = mRight.y;
mView(2, 0) = mRight.z;
mView(3, 0) = x;

// mUp:摄像机正上方 -- 观察坐标系+y轴
mView(0, 1) = mUp.x;
mView(1, 1) = mUp.y;
mView(2, 1) = mUp.z;
mView(3, 1) = y;

// mLook:摄像机正前方 -- 观察坐标系+z轴
mView(0, 2) = mLook.x;
mView(1, 2) = mLook.y;
mView(2, 2) = mLook.z;
mView(3, 2) = z;

mView(0, 3) = 0.0f;
mView(1, 3) = 0.0f;
mView(2, 3) = 0.0f;
mView(3, 3) = 1.0f;

mRight是观察坐标系+x轴的向量在世界坐标系中的坐标表现 -> 观察坐标系是框架A,世界坐标系是框架B,所以得到的是从观察坐标系转换到世界坐标系的坐标转换矩阵 -- 所以需要求逆

8.DirectXMath库提供的变换函数:

// 1.这些函数的作用是构造(返回)一个变换矩阵
// 构造缩放矩阵
XMMATRIX XM_CALLCONV XMMatrixScaling(float ScaleX,float ScaleY,float ScaleZ); 
// 用3D向量的分量来构造缩放矩阵
XMMATRIX XM_CALLCONV XMMatrixScalingFromVector(FXMVECTOR Scale); 

// 弧度 以顺时针方向旋转
XMMATRIX XM_CALLCONV XMMatrixRotationX(float Angle); // float Angle是以弧度为单位

// 绕旋转轴Axis顺时针旋转
XMMATRIX XM_CALLCONV XMMatrixRotationAxis(FXMVECTOR Axis,float Angle); 

// 平移
XMMATRIX XM_CALLCONV XMMatrixTranslation(float OffsetX,float OffsetY,float OffsetZ);
// 平移 用向量构造
XMMATRIX XM_CALLCONV XMMatrixTranslationFromVector(FXMVECTOR Offset); 


// 2.计算:
// 计算向量与矩阵的乘积vM 针对点的变换 Vw默认为1
XMVECTOR XM_CALLCONV XMVector3TransformCoord(FXMVECTOR V,CXMMATRIX M); 

// 针对向量的变换 Vw默认为0
XMVECTOR XM_CALLCONV XMVector3TransformNormal(FXMVECTOR V,CXMMATRIX M); 

注意:参数float Angle是以弧度为单位(虽然参数名字是angle而不是radian)

其实弧度值根本没必要转换为角度值,弧度制为我们常见的表示形式(\frac{\pi}{3}, \frac{\pi}{4},\frac{\pi}{2}等等),且三角函数的参数都是以弧度制作为单位

弧度与角度的转换公式:radian = \frac{angle \times \pi }{ 180 }

代码示例:

#include <windows.h>
#include <iostream>
#include <DirectXMath.h>
#include <DirectXPackedVector.h>

using namespace std;
using namespace DirectX;
using namespace DirectX::PackedVector;

ostream& XM_CALLCONV operator<<(ostream& os, FXMMATRIX m)
{
	XMFLOAT4X4 dest;
	XMStoreFloat4x4(&dest, m);

	for (int i = 0; i < 4; i++) {
		for (int j = 0; j < 4; j++) {
			os << dest.m[i][j] << " ";
		}
		os << "\n";
	}
	return os;
}

int main()
{
	XMMATRIX m1 = XMMatrixSet(
		1.f, 2.f, 3.f, 0.f,
		0.f, 1.f, 0.f, 0.f,
		0.f, 0.f, 1.f, 0.f,
		2.f, 2.f, 2.f, 1.f
	);
	cout << m1 << endl;

	XMMATRIX m2 = XMMatrixScaling(1.f, 2.f, 3.f);
	cout << m2 << endl;

	XMVECTOR v1 = XMVectorSet(1.f, 2.f, 3.f, 0.f);
	XMMATRIX m3 = XMMatrixScalingFromVector(v1);
	cout << m3 << endl;

	XMMATRIX m4 = XMMatrixRotationX(30 * XM_PI / 180); 
	cout << m4 << endl;

	XMVECTOR v2 = XMVectorSet(1.f, 2.f, 3.f, 0);
	v2 = XMVector4Normalize(v2);
	XMMATRIX m5 = XMMatrixRotationAxis(v2, 60 * XM_PI / 180);
	cout << m5 << endl;
}

本文含有隐藏内容,请 开通VIP 后查看