Managed DirectX +C# 开发(入门篇)3
Managed DirectX +C# 开发(入门篇)3 Managed DirectX +C# 开发(入门篇)3
第五章 矩阵变换
一:为什么使用4×4矩阵?
在3D中进行编程时,使用的是4 × 4 矩阵来进行矩阵变换。初学者往往认为既然是三维空间,为何不使用3 × 3 的矩阵来表示呢?
这是因为3 × 3 的矩阵不能表示有些变换,比如比如平移、投影、反射,因此,增大到4 × 4 ,这样,就可以描述更多的变换了。
但是矩阵变成了4 × 4 ,为了做向量与矩阵的相乘,所以需要把向量增加为1 × 4 ,因为1 × 3 的行矩阵和4 × 4 的矩阵是无法相乘的;
那么,如何使用第四个成员(用w 来表示)呢?当把一个点放置到一个1×4的行矩阵中时,设置w =1,表示可以对点进行适当的平移,如果w =0,表示不可以对向量进行平移。
比如:把点p = (1, 2, 3)放置到一个行向量中就象这样[1, 2, 3, 1],同样把向量v = (2 1, 3 2, 3 3) 放置到一个行向量中就象这样[2 1, 3 2, 3 3, 0];
二、矩阵平移
对图形进行平行移动是最基本的操作:
使用下面的矩阵可以把向量(x, y, z, 1)
沿x轴移动px个单位,沿y轴移动py
个单位,沿z轴移动pz
个单位:
以下代码为设置一个向量,对它应用一个平移变换后,得到新的向量,完整的代码参见程序:
private void button1_Click(object sender, System.EventArgs e)
{
Clear();
// 定义一个测试的向量;
Vector3 myVect=new
Vector3(3,6,8);
// 用此向量来保存后来的结果;
Vector4 resultVect =new
Vector4();
// 生成一个平移矩阵,XYZ方向均增加4个单位
Matrix MatrixA=Matrix.Translation(4,4,4);
// 对向量应用平移变换;
resultVect=Vector3.Transform(myVect,MatrixA);
// 将返回的向量显示出来;
label9.Text =resultVect.ToString();
// 将矩阵也显示出来;
DisplayMatrix(MatrixA,label1);
}
程序执行结果,执行程序,点击按钮,执行结果如下:
下面应用一个具体的实例:
1:新建一个工程,生成一个XOZ水平面(代码略):
2:定义二个Mesh对象:Mesh mesh1=null,mesh2=null;
3:在图形的初始化函数中对它进行实例化,在这里是生成两个半径一样大小的小球:
public bool InitializeGraphics()
{
……
mesh1=Mesh.Sphere(device,0.5f,20,20);
mesh2=Mesh.Sphere(device,0.5f,20,20);
…….
}
4:在渲染场景函数里:一个小球进行平移变换,另一个则不进行任何变换:
private void Render()
{
if
(device == null
)
return
;
device.Clear(ClearFlags.Target, System.Drawing.Color.Black , 1.0f, 0);
SetupCamera();
device.BeginScene();
displayPlane();
device.Transform.World=Matrix.Translation(10,0,0);
mesh1.DrawSubset(0);
device.Transform.World=Matrix.Identity;
mesh2.DrawSubset(0);
device.EndScene();
device.Present();
}
程序执行结果如下:
三、矩阵旋转
如果对一个图形绕一个固定点或一条线进行旋转,结果示意如下:
图9
下面的矩阵把一个向量围绕x,y 和z轴旋转一个角度。注意这里应当为弧度值, 当俯视绕轴原点时,角度是指顺时针方向的角度。
示例一:
下面测试一向量应用旋转变换后的结果:
private
void
button2_Click(object
sender, System.EventArgs e)
{
Clear();
Vector3 myVect=new
Vector3(3,6,8);
Vector4 resultVect =new
Vector4();
Matrix MatrixA=Matrix.RotationX((float
)Math.PI/2);
resultVect=Vector3.Transform(myVect,MatrixA);
label9.Text =resultVect.ToString();
DisplayMatrix(MatrixA,label1);
}
在这里,我们是让此向量绕X轴旋转90度,可以想像成直线OA绕X轴旋转90度,得到直线OB,B点值即为所求向量值;
执行结果如下:
示例二:
下面应用一个具体的实例:
1:新建一个工程,生成一个XOZ水平面(代码略):
2:定义二个Mesh对象:Mesh mesh1=null,mesh2=null;
3:在图形的初始化函数中对它进行实例化,在这里是生成两个长方体:
public bool InitializeGraphics()
{
……
mesh1=Mesh.Box(device,0.5f,0.5f,16);
mesh2=Mesh.Box(device,0.5f,0.5f,16);
……
}
4:在渲染场景函数里:对一个长方体绕X轴旋转45度,另一个长方体则在原点绘出:
private void Render()
{
……
device.Transform.World=Matrix.RotationX((float
)Math.PI/4);
mesh1.DrawSubset(0);
device.Transform.World=Matrix.Identity;
mesh2.DrawSubset(0);
……
}
程序执行结果如下:
以上介绍的是绕X轴旋转,绕Y,Z轴旋转类似,矩阵是:
四、矩阵缩放
对物体进行缩放的示意图如下:
下面的矩阵即为把向量沿x轴缩放qx个单位,沿y轴缩放qy
个单位,沿z轴缩放qz
个单位:
注意,如果在每个方向上的值大于1为放大,小于1则为缩小;
示例一:
private
void
button3_Click_1(object
sender, System.EventArgs e)
{
Clear();
Vector3 myVect=new
Vector3(3,6,8);
Vector4 resultVect =new
Vector4();
Matrix MatrixA=Matrix.Scaling(2,0.5f,0.5F);
resultVect=Vector3.Transform(myVect,MatrixA);
label9.Text =resultVect.ToString();
DisplayMatrix(MatrixA,label1);
}
代码中,将向量X方向扩大为2倍,Y和Z方向分别缩小为2倍;
得到结果向量值为:
示例二:
以下代码将前面创建的两个长方体其中一个应用缩放变换,其中在X,Y方向上放大为原来的2倍,Z方向上缩小为原来的2倍,相关代码如下:
…… device.Transform.World=Matrix.Scaling(2,2,0.5f)*Matrix.RotationX((float
)Math.PI/4);
mesh1.DrawSubset(0);
device.Transform.World=Matrix.Identity;
mesh2.DrawSubset(0);
……
程序执行结果如下:
五、组合变换
一般情况下,需要对向量进行一系列的变换。比如,前面显示的长方体,为了不使两者叠在一起,先进行了缩放,再进行了旋转,其实还可以再应用其它变换,比如平移;
假设把向量p = [5, 0, 0, 1] 在所有轴上缩小为原来的1/5,然后沿着y轴旋转π/4,最后把它在x轴上移动1个单位,在y轴上移动2个单位,在z轴上移动3个单位。
设缩放、旋转、移动的变换矩阵分别是 S , R y, T ,如下:
因此:
这样就得到了最后的结果,另一种简便的方法是通过使用矩阵相乘把3个变换矩阵合成一个矩阵。注意次序不要颠倒,否则结果会不一样。
那么 pQ = [1.707, 2, –3.707, 1]。
和第一种方法得到的结果是一样的;
六、其它常用变换
有两个矩阵,常常用来设置场影及摄像机:
1:device.Transform.Projection用来获取或者设置投影变换矩阵,它本身也是一个Matrix型的变量。
2:Matrix.PerspectiveFovLH方法:
public static Matrix PerspectiveFovLH(
float fieldOfViewY, //Y方向的视觉范围,弧度表示,即Y方向的视域角
float aspectRatio, //视觉范围的长宽比
float znearPlane, //近裁剪平面
float zfarPlane//远裁剪平面
);
定义长宽比的作用是,如果用户改变了窗口的大小,那么仍然就可以正确的显示物体的大小。最后两个参数是远及近裁剪平面,必须为这两个平面指定2个Z值,一般来说,假设物体将在Z轴的1.0f和1000f之间运动时,就分别把这两个参数设为1.0f,1000.0f。
3:Matrix.LookAtLH方法。 在使用观察矩阵来建立一个立体的照相机模型后,需要设置不同的观察点,可以使用这种方法来设置相机。
public static Matrix LookAtLH(
Vector3 cameraPosition,//相机位置
Vector3 cameraTarget, //相机的方向
Vector3 cameraUpVector //相机的Up向量
);
其中第二个参数为相机的面对方向,假设相机面对的是Z轴的方向,那么就设置为(0,0,1)。