OpenGL学习脚印: 坐标和变换的数学基础(math-coordinates and transformations)

本文主要是介绍OpenGL学习脚印: 坐标和变换的数学基础(math-coordinates and transformations),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

写在前面
上一节介绍了向量和矩阵,本节将熟悉坐标、线性变换、仿射变换以及坐标转换等概念和计算方法,这些内容对后续的学习将会有很大帮助。部分内容不是OpenGL编程初学者所必须掌握的,可以在以后需要时再回头来看。

这里是对这些知识点的一个总结,旨在对他们有个整体把握,后面具体应用时会使用这些概念。内容尽量以例子形式说明,仅在必要时会给出数学证明。一个主题往往涉及过多内容,对于文中省略的部分,请参考相应的教材。

通过本节可以了解到

  • 坐标的概念
  • 线性变换的概念和计算方法
  • 仿射变换
  • 坐标转换的概念和计算方法

坐标系及坐标

坐标是在特定坐标系下表示物体位置的方法,一谈到坐标,必定是在某个坐标系下给定的。例如经纬度坐标是相对于地球的球面坐标系统给出的。

坐标系则给出了一个参考框架,在这个框架里面,定义其他位置相对于一个起始点(这个起始点称作原点 O )的位置。同一位置,在不同的坐标系下会有不同的坐标,例如你所在城市以经纬度坐标表示,是相对于地球坐标系来给定的,如果从太阳系或者银河系来给定,又会是另外一个不同的值。
常见的坐标系包括:笛卡尔坐标系,极坐标系,球面坐标系等。
如下图所示的我们熟悉的,2D笛卡尔坐标系(来自wiki):
这里写图片描述

如下图所示的熟悉的3D笛卡尔坐标系(来自wiki):
这里写图片描述

定义坐标系

定义一个坐标系需要指定(参考自Objects in Motion):

  • 坐标系的维度 2D, 3D, 4D等等
  • 定义坐标空间轴的向量 这些向量成为基向量,他们有名字,例如x,y,z;这些向量一般而言都是正交的,但不一定非得互相正交(只要线性无关即可,后面介绍),但是每一个维度必须只有一个轴。
  • 坐标系的原点O 原点是导出其他点的参考点。
  • 一个表明空间中点是否合法的区域 在此区域之外的点就不再合法。这个区域根据空间不同,可以是无穷的。

这里,维度已由基向量维数确定,合法区域一般是无穷的,但是在图形处理中某些坐标空间是有限的,例如规范化设备空间(后面其他文章会介绍)。作为一个了解,基向量不一定非得正交,如下图所示:
非正交的轴

对于一般情况,我们只需要记住:
坐标系=(基向量,原点O)

左手坐标系和右手坐标系

对于任意2个2D坐标系,我们通过旋转、移动翻转可以将两个坐标系xy轴的指向相同。

但是对于3D坐标系,任意两个坐标系却不能等价。实际上,存在两种完全不同的3D坐标系:左手坐标系和右手坐标系。如果同属于左手坐标系或者右手坐标系,则可以通过旋转来重合,否则不可以。
判断一个坐标系是否属于右手系,可以拿出右手,然后右手的大拇指代表+x轴指向,食指代表+y轴指向,中指代表+z轴指向,你可以转动你的右手来匹配这个坐标系,如果能匹配则是右手坐标系,否则是左手坐标系。判断左手坐标系的方法类似。如下图所示为判断右手坐标系的方法(来自OpenGL coordinate system):
这里写图片描述

关于左右手坐标系理解还可以可参考下图(来自《3D数学基础》):
左手右手坐标系
如上右图右手坐标系,这里拇指、食指、其余手指分别代表x,y,z轴的正方向。高等数学教材上使用的是右手坐标系。

旋转正方向的判断

同样还存在一个左手规则和右手规则,可以用于判断当物体绕轴旋转时正方向的判定问题。

对于左手规则,确定一个旋转轴后,左手握成拳头,拇指指向旋转轴的正方向,则其余手指弯曲方向即为旋转的正方向。从旋转轴正向末端来看,正向旋转是顺时针的。对于右手规则,有同样的方法。可参见下图:
左手右手规则

左手右手规则在不同场合有着不同作用。上一节,我们使用右手规则判断了叉积的结果向量的方向。

注意OpenGL中坐标系 OpenGL中的物体、世界、照相机坐标系都属于右手坐标系,而规范化设备坐标系使用左手坐标系。笼统地说OpenGL使用右手坐标系是不合适的。这些坐标系后面会介绍。关于这个问题可以参考SO.

坐标

坐标是在指定坐标系中,相对于原点 O 给出的一个位置。这个位置可以用有序实数对表示(有些坐标系中可能使用复数),注意数对中的数字顺序对结果有影响。上面提到坐标系=(基向量,原点O),3D坐标系用向量表示为:
i=(ax,by,bz)j=(bx,by,bz)k=(cx,cy,cz)()
O=(Ox,Oy,Oz)()
这样在坐标系中一点 P 与原点O构成的向量:
r=OP=xi+yj+zk
这时称 (x,y,z) 为点 P 的坐标,这个坐标也可以表示向量OP.
一般地使用的3D笛卡尔坐标系使用标准基向量和坐标原点:
i=(1,0,0)j=(0,1,0)k=(0,0,1)()
O=(0,0,0)() .

从上面可以看到,在一个坐标系中,求取坐标的过程,是一个向量分解的过程。求取一个位置在另一个不同的坐标系中的坐标,则需要进行坐标转换。后面会介绍。

使用坐标系的优势

使用坐标系统便能以解析几何的形式来研究空间几何。通过建立一个坐标系使得空间中点用有序实数组表示,空间图形用方程表示,这样能方便地研究几何图形的性质。


必要基础概念

了解线性变换、仿射变换以及坐标转换,对于后面学习图形编程中的模型变换方法有很大帮助,因此这里予以介绍。要了解这些概念,需要一些其他概念的支撑,这里逐一介绍。每一个概念以定义结合示例的形式给出,如果暂时没有理解清楚,可以暂时跳过,以后回过头来再看或者参考这个主题的其他资料。本部分最后会给出完整的线性变换示例。

向量组的线性组合

向量组是一组向量的集合,例如 α1,α2,,αm 表示一个由m个 n×1 的矩阵(n维列向量)组成的列向量组。对应的也有行向量组的概念。如果存在一组实数 λ1,λ2,,λm ,使得向量 β 满足下式:
β=λ1α1+λ2α2++λmαm(1)

则称向量 β α1,α2,,αm 的线性组合,或者说 β α1,α2,,αm 线性表示。上述求解点 P 坐标时向量r分解为向量组 i,j,k 表示的过程,就是一个找出线性表示系数的过程。

向量组线性无关

对于向量组 α1,α2,,αm ,如果存在不全为零的数 λ1,λ2,,λm ,使下面的等式成立:
λ1α1+λ2α2++λmαm=0(2)
则称向量组 α1,α2,,αm 线性相关(linearly dependent), 否则称为线性无关( linearly independent)。也就是要使向量组 α1,α2,,αm 线性无关,那么所有的系数 λi 都必须为0。

线性相关的一种几何解释 来自math.stackexchange:
假定你有一组向量 {x1,x2,,x5} ,你从某个点出发,沿着 x1 走动一段距离,然后沿着 x2 走动一段距离, 最后沿着 x5 走动一段距离,最终你又回到了出发点(这里表明存在 λ1x1+λ2x2+λ5x5=0 ,即式2成立)。这就说明 {x1,x2,,x5} 是线性相关的。

当然存在其他方法,判断向量组线性相关性,感兴趣的可以参考线性代数教材。

线性空间的基

向量空间(也叫做线性空间)是对我们经常使用的2D和3D空间一般规律的拓展,它的定义主要反映的是满足一系列的运算规律,例如交换律和结合律等。由于这个定义包含较多规则,不在此列出,感兴趣的可以参考vector space。

如果在线性空间V中存在n个线性无关的向量 α1,α2,,αn 使得V中任意元素 α 都能由他们线性表示,则称 α1,α2,,αn 为V的一个。基所含向量个数n称为线性空间V的维数,并称V为n维线性空间。

例如2D空间中,二维向量组 i=(1,0),j=(0,1) 是它的一个基;在3D空间中,向量组: i=(1,0,0),j=(0,1,0),k=(0,0,1) 是它的一个基。类似的可以推广到n维向量空间的基。

向量在基下的坐标

α1,α2,,αn 是n维线性空间V的一个基,若任取 αV ,总有且仅有一组有序实数 x1,x2,,xn ,使得:
α=x1α1+x2α2++xnαn=(α1,α2,,αn)x1x2xn(3)
成立,则称这组有序数 x1,x2,,xn 为元素 α 在基 α1,α2,,αn 下的坐标,记作 (x1,x2,,xn)T

这里元素 α 用基向量组的线性组合来表示,坐标就是线性组合的系数。例如向量 a=(3,4)=3e1+4e2 ,其中 e1=(1,0),e2=(0,1) 为标准基,则a的坐标为 (3,4)T


线性变换(Linear Transformations)

变换这个词类似于函数,即将一个定义域里的输入量转化为值域里的另一个值,变换就是一个映射关系,一种规则。线性变换的一些性质对于后续学习3D模型变换时,理解起来将会更容易。

线性变换 T:UV 是一个函数,将定义域U中元素,映射到值域V中,并满足下列两个条件(参考Definition LT):
1)可加性 对任意 u1,u2U ,都满足: T(u1+u2)=T(u1)+T(u2)
2)齐次性 对任意 uU 和任意标量 k ,都满足: T(ku)=kT(u)

可以利用上面的两个条件,即可加性和齐次性条件验证一个变换是否是线性变换。
线性变换示例(来自Example ALT)
T(x1x2x3)=[2x1+x34x2]
首先验证其是否满足可加性:

T(x+y)=T(x1x2x3+y1y2y3)=(x1+y1x2+y2x3+y3)=[2(x1+y1)+(x3+y3)4(x2+y2)]=[2x1+x34x2]+[2y1+y34y2]=T(x1x2x3)+T(y1y2y3)=T(x)+T(y)(*)

然后验证是否满足齐次性:

T(kx)=T(kx1x2x3)=T(kx1kx2kx3)=[2(kx1)+(kx3)4(kx2)]=k[2x1+x34x2]=kT(x1x2x3)=kT(x)(**)

由上述*和**式可知,变换T满足可加性和齐次性,因而是线性变换。
实际上判断线性变换还有一个 简单方法,即看变换后每个组成部分是不是原来分量的倍数(不能为常量)。
例如变换 f(x,y,z)=(3xy,3z,0,z2x) 是线性变换,而:
g(x,y,z)=(3xy,3z+2,0,z2x)
h(x,y,z)=(3xy,3xz,0,z2x)
都不是线性变换。其中g中 3z+2 的部分2是个常量,不满足倍数要求;h中 3xz 不满足倍数要求,因而也不是线性变换。
对于线性变换一个重要的性质是(令齐次性表达式中u=0可得到):
T(0)=0(1)

线性变换与矩阵一一对应

对一个线性变换 T ,存在一个矩阵A与之对应,变换表示为 T(x)=Ax ,其中 x 为列向量。

证明: 1.首先证明充分性。
T是线性变换时,对于 xU ,其中

x=x1x2xn
,有:
T(x)=T(x1e1+x2e2++xnen)=x1T(e1)+x2T(e2)++xnT(en)=(T(e1),T(e2),,T(en))x1x2xn=Ax

其中:
A=(T(e1),T(e2),,T(en)) 称为标准矩阵,它的每一列都是标准基 ei 变换后的向量在原基下的表示。
2.证明必要性
当变换 T 对应矩阵A,表示为 T(x)=Ax 时,证明 T 为线性变换。
由矩阵乘法性质,容易得到:
T(x+y)=A(x+y)=Ax+Ay=T(x)+T(y)()
T(kx)=A(kx)=kA(x)=kT(x)()
则说明 T 满足可加性和齐次性,因而是一个线性变换。
由1,2得到,线性变换T总是与一个矩阵 A 一一对应。

例如矩阵A如下(下面的例子来自: Matrices and linear transformations):
A=[130112].
对应的线性变换为:
T(x)=Ax=[130112]xyz=[xz3x+y+2z]=(xz,3x+y+2z).
由给定的线性变换:
f(x,y)=(2x+y,y,x3y)
要找到对应的矩阵A,由上面定理证明时可知,标准矩阵A的内容为:
A=(f(e1),f(e2)))
其中 f(e1) 计算如下:
f(e1)=f(1,0)=(2,0,1)=201.
f(e2) 计算如下:
f(e2)=f(0,1)=(1,1,3)=113.
所以最终计算得到矩阵 A 如下:
A=201113.

后面会详细介绍矩阵A的计算方法。在此之前,先来看下线性变换的应用。

线性变换的应用

线性变换在3D图形中模型变换部分应用很多,例如旋转、错切(shear)、缩放等操作都是线性变换。在OpenGL中使用矩阵操作来表示这些线性变换。例如2D平面上绕原点的旋转 θ 角度的操作如下图所示:
旋转

通过利用极坐标系表示 x,yx,y 并利用三角函数公式(具体过程略,有兴趣可以自行推导,后面会介绍另外一个更简洁的方法),得到旋转矩阵为:
[xy]=[cosθsinθsinθcosθ][xy]
那么矩阵:
A(θ)=[cosθsinθsinθcosθ] 表示的就是这个围绕原点旋转 θ 角度的线性变换。

使用矩阵表达变换的优势 通过使用矩阵形式,能够级联对物体顶点的变换,例如先执行旋转,后执行缩放,则表示为: T(x)=TsTrx=(TsTR)x ,可以在对描述物体组成的多个顶点执行变换之前,先计算出矩阵 (TsTR) ,从而节省执行变换需要的时间;同时也能表示为逆操作,例如旋转 θ 角度, T(x)=A(θ)x 。只是在OpenGL中使用的标准矩阵 A 是4x4的,后面会介绍原因和具体构造方法。

线性变换对应矩阵的计算方法(重要结论)

线性变换由基及变换后基的值唯一确定,通过计算线性变换后基的值可以得到线性变换对应的矩阵A。 这是本节讲述线性变换最重要的结论。

定理: 设u1,u2,,un是线性空间U的一个基,线性空间V包含向量 v1,v2,,vn (可以相同)。那么存在唯一的线性变换 T:UV 使得: T(ui)=vi(1in)

上面的定理证明,感兴趣的可以参考 Theorem LTDB,不在此处给出详细过程。

这个定理告诉我们: 只要知道了线性空间U的给定基 u1,u2,,un 在线性变换 T 下对应的值T(u1),T(u2),,T(un),线性变换 T 也就由ui T(ui) 的对应关系确定了。
这个定理的作用就好比,两点确定一条直线。因此我们可以通过计算:
A=(T(u1),T(u2),,T(un)) 来获取线性变换 T 对应的矩阵A

也就是说,矩阵 A 的列向量,由T(ui)在基 u1,u2,,un 下的坐标唯一确定。同时给出一个矩阵 A 作为线性变换T在基 u1,u2,,un 下的矩阵,也就给出了该基在线性变换 T 下对应的值T(u1),T(u2),,T(un),从而确定了线性变换 T 。这表明线性变换T与矩阵之间存在一一对应关系,上面已经证明了这个结论。

例如对于上面的旋转矩阵,从基和转换后基的角度,也就是从x,y轴来看, x 转换后为x y 转换后为y,如下图所示(doitpoms.ac.uk):。
这里写图片描述
利用三角函数公式可以计算转换后 x,y 对应的坐标,如下图所示(来自:Rotations and Infinitesimal Generators):
这里写图片描述
因此可以顺利的写出旋转矩阵为:
R(θ)=[cosθsinθsinθcosθ]
矩阵第一列即为 x 转换后的x轴的对应的坐标,第二列即为 y 转换后的y轴对应的坐标。显然这个计算方法,比利用极坐标公式来得快。
在后面使用OpenGL模型变换矩阵时,经常要使用到这个方法,理解了这一点后面理解模型变换矩阵就会变得简单。


基变换与坐标变换(重要结论)

一个位置,在不同的坐标系里有不同的坐标。设 u={u1,u2,,un} v={v1,v2,,vn} 是n维线性空间V的两个基。两个基之间可以互相表示。因此存在9个标量 λij ,使得:
u1=λ11v1+λ12v2+λ13v3
u2=λ21v1+λ22v2+λ23v3
u3=λ31v1+λ32v2+λ33v3
λij 写成矩阵形式得到:

M=λ11λ21λ31λ12λ22λ32λ13λ23λ33

则上面的式子可以简写为:
u=Mv(a)
矩阵 M 称为过渡矩阵,通过矩阵M我们可以把向量在基u下的表示转换为基v下的表示,通过 M1 可以把向量在v下的表示转换为u下的表示。
设有一个向量 w ,它在基u和v下的表示分别为:
w=a1v1+a2v2++anvn=aTv=b1u1+b2u2++bnun=bTu

其中:
a=a1a2an

b=b1b2bn

由上式得到:
aTv=bTu(b)

将式子a代入式子b得到:
a=MTb(c)
b=(MT)1a=Ta(d)

式子c和d给出了向量 w 的两个坐标之间的转换公式。
上面的矩阵MT

MT=λ11λ12λ13λ21λ22λ23λ31λ32λ33

这个矩阵的三个列向量,就是基 u={u1,u2,,un} 的向量在基v下的坐标表示,即:
MT=([u1]v,[u2]v,,[un]v)
对应的 (MT)1 矩阵即是 v={v1,v2,,vn} 在u下的坐标表示。
(MT)1=([v1]u,[v2]u,,[vn]u) .
那么上面的式子c和d可以分别改写为:
[ω]v=[u]v[ω]u(e)
[ω]u=[v]u[ω]v(f)

从式子e和f来看,我们可以得出,要在两个基之间转换坐标,只需要求出一个基在另一个基里的坐标表示即可,这个坐标表示构成矩阵 MT 或者 (MT)1

示例
已知基 u 为:
u1=(1,2,1,0),u2=(3,3,3,0),u3=(2,10,0,0),u4=(2,1,6,2)
v 为:
v1=(1,2,1,0),v2=(1,1,1,0),v3=(1,0,1,0),v4=(0,0,0,2)
容易求出:
v1=u1
v2=2u1+u2
v3=11u14u2+u3
v4=27u1+11u22u3+u4
即:

[v]u=1000210011410271121

也可得:
u1=v1
u2=2v1+v2
u3=3v1+4v2+v3
u4=v13v2+2v3+v4
即:

[u]v=1000210034101321

给定向量 ω=(6,1,2,2) ,容易验证:
ω=v1+3v2+2v3+v4
也就是:
[w]v=1321
那么:

[w]u=[v]u[w]v=10002100114102711211321=10601

容易验证:
(1) w=10u1+6u2+u4
(2) [u]1v=[v]u


仿射变换

线性变换无法表达一类重要的变换——平移变换。平移变换表达的是对于点 p=(x,y,z) 经过 d=(αx,αy,αz) 所表示的位移后得到点 p=(x,y,z) 的过程,表示为:
p=p+d
我们尝试寻找变换 T 满足:T(x)=Ax=p=x+αxy+αyz+αz
d0 时,上式中 T(0)0 ,由式子1可知,这不是线性变换。因此需要引入仿射变换的概念。

仿射变换与线性变换不同之处在于,线性变换保持原点位置不变,而仿射变换可以改变原点的位置。仿射变换包括线性变换,例如旋转,缩放等变换,特殊地是仿射变换包括平移变换。例如 f(x)=2x 是一个线性变换,也是仿射变换;而 f(x)=2x+3 是仿射变换。感兴趣的可以参考What is the difference between linear and affine function。一般而言,仿射变换是 线性变换+平移变换。

使用3x3矩阵无法表达平移变换(当 d0 时方程 T(x)=A3×3x=p 无解,感兴趣可以参考Reason for homogeneous),但是以一个统一的方式表达变换,在计算中将便于计算,因此需要引入齐次坐标系的概念来完成这个目标。从编程角度来讲,在OpenGL中引入齐次坐标系主要是为了表达平移变换和投影变换中的透视除法。如果你要从数学角度理解齐次坐标系,这对数学要求较高,将会涉及到射影几何等概念,感兴趣的话可以参考The Truth Behind Homogeneous Coordinates。

齐次坐标(homogeneous coordinate)

齐次坐标是在原来坐标的基础上添加了一个 w 成分。在3D中使用(x,y,z)既能表示点,又能表示向量,容易引起混淆。使用齐次坐标能够克服这个困难。具体做法是,在由原点 P0 和基 v1,v2,v3 所定义的坐标系中,任意点 P 可以表示为:
P=α1v1+α2v2+α3v3+P0
定义标量0和1与点的乘法为:
0.P=0,1.P=P ,则可以把点P表示为:
P=[α1α2α31]v1v2v3P0
可以把向量 w=β1v1+β2v2+β3v3 表示为:
w=[β1β2β30]v1v2v3P0
因此,使用齐次坐标系表示点形式为:
P=α1α2α31
表示向量为:
w=β1β2β30

这里齐次坐标的 w 成分为0或者1,在后面投影变换的透视除法时还会看到w成分的其他值。

使用4x4矩阵表示仿射变换

对于上面的平移变换:
T(x)=Ax=p=x+αxy+αyz+αz
A 为4x4矩阵时,使用齐次坐标,x表示为p=(x,y,z,1), d=(αx,αy,αz,0) , p=(x,y,z,1)
可以计算得到 A 为:

A=100001000010αxαyαz1
.
上面的矩阵A对应平移变换,对于线性变换,矩阵A中:
d=(αx,αy,αz,0)=(0,0,0)
也就是说矩阵A的结构为:
这里写图片描述

使用4x4矩阵表示坐标转换

前面介绍的基变换与坐标变换,没有考虑到原点,两个坐标系的原点位置可能不同,因此需要使用4x4矩阵调整下上面的结果。对于参考点 P0 和基 u={u1,u2,,un} ,以及参考点 Q0 和基 v={v1,v2,,vn} ,除了上面的等式:
u1=λ11v1+λ12v2+λ13v3
u2=λ21v1+λ22v2+λ23v3
u3=λ31v1+λ32v2+λ33v3
还需要添加一个 Q0 P0 关系式:
P0=Q0+λ41v1+λ42v2+λ43v3
写成矩阵形式为:
u1u2u3P0=Mv1v2v3Q0

其中M是一个4x4矩阵:

M=λ11λ21λ31λ41λ12λ22λ32λ42λ13λ23λ33λ430001

坐标转换的其余部分,与上面讲述的3x3矩阵类似。

坐标转换示例1

下图所示的旋转坐标系后,求取原坐标系下点p在[s,t]坐标系下坐标,如下图所示:
旋转
直接利用极坐标推导的过程,请参考 坐标旋转变换公式的推导。我们使用上面介绍的寻找基向量之间关系的方法来计算这个变换矩阵。
已知 xoy 坐标系下基 e=(e1,e2) sot 坐标系的基表示矩阵为:

[e]sot=[cosθsinθsinθcosθ]

那么:
[w]sot=[e]sot[w]xoy
即:
[st]=[cosθsinθsinθcosθ][xy]

坐标转换示例2

这个例子来自参考资料[1]。
如下图所示,存在两个坐标系,世界坐标系和照相机坐标系:
这里写图片描述

其中照相机坐标系的原点位于世界坐标系的 P(1,0,1,1) 处,并且照相机指向世界坐标系的原点,则照相机指向向量 n=(1,0,1,0)T ,照相机的观察正向同世界坐标系地y轴一样,为 v=(0,1,0,0)T ,利用叉积 n×v 计算出相机的第三个正交的方向 u=(1,0,1,0)T
那么点O和xyz轴构成的世界坐标系和以点P及uvn构成的照相机坐标系之间存在转换关系。其中uvnP在世界坐标系的表示为:

[uvnP]xyzO=1010010010101011

xyzO在照相机坐标系下的表示为:
[xyzO]uvnP=[uvnP]1xyzO=1/201/2001001/201/200011

则世界坐标系的原点O在uvnP坐标系下的坐标为:

[O]uvnP=[xyzO]uvnP[O]xyzO=1/201/2001001/201/2000110001=0011

这个即求出世界坐标系的原点在照相机坐标系下的 (0,0,1,1) 处。

同时照相机坐标系下的原点P在世界坐标系 xyzO 下的表示为:

[P]zyzO=[uvnP]zyzO[P]uvnP=10100100101010110001=1011

这个即求出了照相机坐标系的原点在世界坐标系的 P(1,0,1,1) 处。
这个例子说明了不同坐标系下坐标之间的转换,后面还会详细介绍OpenGL中使用的坐标转换。

参考资料

[1]《交互式计算机图形学-基于OpenGL着色器的自动向下方法》电子工业出版社 Edward Angle等著
[2]《3D数学基础:图形与游戏开发》 清华大学出版社
[3]《线性代数》武汉大学数学与统计学院 高等教育出版社 齐民友主编
[4] Linear Transformations
[5] matrices_linear_transformations
[6] Linear transformations
[7] Objects in Motion
[8] 很棒的 LINEAR TRANSFORMATIONS 课件

这篇关于OpenGL学习脚印: 坐标和变换的数学基础(math-coordinates and transformations)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/226877

相关文章

使用C#代码计算数学表达式实例

《使用C#代码计算数学表达式实例》这段文字主要讲述了如何使用C#语言来计算数学表达式,该程序通过使用Dictionary保存变量,定义了运算符优先级,并实现了EvaluateExpression方法来... 目录C#代码计算数学表达式该方法很长,因此我将分段描述下面的代码片段显示了下一步以下代码显示该方法如

MySQL中my.ini文件的基础配置和优化配置方式

《MySQL中my.ini文件的基础配置和优化配置方式》文章讨论了数据库异步同步的优化思路,包括三个主要方面:幂等性、时序和延迟,作者还分享了MySQL配置文件的优化经验,并鼓励读者提供支持... 目录mysql my.ini文件的配置和优化配置优化思路MySQL配置文件优化总结MySQL my.ini文件

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

Ilya-AI分享的他在OpenAI学习到的15个提示工程技巧

Ilya(不是本人,claude AI)在社交媒体上分享了他在OpenAI学习到的15个Prompt撰写技巧。 以下是详细的内容: 提示精确化:在编写提示时,力求表达清晰准确。清楚地阐述任务需求和概念定义至关重要。例:不用"分析文本",而用"判断这段话的情感倾向:积极、消极还是中性"。 快速迭代:善于快速连续调整提示。熟练的提示工程师能够灵活地进行多轮优化。例:从"总结文章"到"用

【前端学习】AntV G6-08 深入图形与图形分组、自定义节点、节点动画(下)

【课程链接】 AntV G6:深入图形与图形分组、自定义节点、节点动画(下)_哔哩哔哩_bilibili 本章十吾老师讲解了一个复杂的自定义节点中,应该怎样去计算和绘制图形,如何给一个图形制作不间断的动画,以及在鼠标事件之后产生动画。(有点难,需要好好理解) <!DOCTYPE html><html><head><meta charset="UTF-8"><title>06

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;

usaco 1.2 Transformations(模拟)

我的做法就是一个一个情况枚举出来 注意计算公式: ( 变换后的矩阵记为C) 顺时针旋转90°:C[i] [j]=A[n-j-1] [i] (旋转180°和270° 可以多转几个九十度来推) 对称:C[i] [n-j-1]=A[i] [j] 代码有点长 。。。 /*ID: who jayLANG: C++TASK: transform*/#include<

零基础学习Redis(10) -- zset类型命令使用

zset是有序集合,内部除了存储元素外,还会存储一个score,存储在zset中的元素会按照score的大小升序排列,不同元素的score可以重复,score相同的元素会按照元素的字典序排列。 1. zset常用命令 1.1 zadd  zadd key [NX | XX] [GT | LT]   [CH] [INCR] score member [score member ...]

【机器学习】高斯过程的基本概念和应用领域以及在python中的实例

引言 高斯过程(Gaussian Process,简称GP)是一种概率模型,用于描述一组随机变量的联合概率分布,其中任何一个有限维度的子集都具有高斯分布 文章目录 引言一、高斯过程1.1 基本定义1.1.1 随机过程1.1.2 高斯分布 1.2 高斯过程的特性1.2.1 联合高斯性1.2.2 均值函数1.2.3 协方差函数(或核函数) 1.3 核函数1.4 高斯过程回归(Gauss

uva 10014 Simple calculations(数学推导)

直接按照题意来推导最后的结果就行了。 开始的时候只做到了第一个推导,第二次没有继续下去。 代码: #include<stdio.h>int main(){int T, n, i;double a, aa, sum, temp, ans;scanf("%d", &T);while(T--){scanf("%d", &n);scanf("%lf", &first);scanf