學習Shader所需的數學基礎(矩陣)

在三維數學中,咱們一般會使用矩陣來進行變換。一個矩陣能夠把一個矢量從一個座標空間轉換到另外一個座標空間。函數

矢量和矩陣

咱們能夠用矩陣來表示矢量。實際上,矢量能夠當作是n×1的列矩陣或1×n的行矩陣,其中n對應了矢量的維度。
把矢量和矩陣聯繫在一塊兒的緣由是爲了讓矢量能夠像一個矩陣同樣參與矩陣的運算。學習

矩陣的運算

矩陣和標量的乘法

矩陣和標量相乘,結果仍然是一個相同維度的矩陣
它們之間的乘法很是簡單,就是矩陣的每一個元素和該標量相乘。spa

公式

4034e8bf-d463-4c8b-903c-b3b85409b08f.jpg

矩陣和矩陣的乘法

一個r×n的矩陣A和一個n×c的矩陣B相乘,它們的結果AB將會是一個r×c大小的矩陣。這意味着,矩陣相乘,第一個矩陣的列數必須和第二個矩陣的行數相同。3d

公式

設兩個矩陣A和B相乘的結果是矩陣C,那麼,C中的每個元素Cij等於A的第i行所對應矢量和B的第j列所對應矢量的點積
74b1c181-1da4-4695-8a1b-4eff1f3c5b62.jpg
以一個簡單的方式解釋:對於每一個元素Cij,咱們找到A中的第i行和B中的第j列,而後把他們的對應元素相乘後再加起來,這個和就是Cijcode

性質

  • 矩陣乘法不知足交換律
  • 矩陣乘法知足結合律

ae0c0fea-295b-4320-96c2-cd1de7af12a3.png

方塊矩陣

方塊矩陣,簡稱方陣,是指那些行和列數目相等的矩陣。在三維渲染裏,最常使用的就是3×3和4×4的方陣
方陣的對角元素指的是行號和列號相等的元素。
若是一個矩陣除了對角元素外的全部元素都爲0,那麼這個矩陣就叫作對角矩陣
一個4×4的對角矩陣:
9fc264f6-434f-47ac-bc0d-b59a4e7179c0.pngblog

單位矩陣

若是一個對角矩陣的對角元素都爲1,那麼這個矩陣被稱爲單位矩陣。一個3×3的單位矩陣以下所示:
78ed8a68-fcf1-4511-86cb-04f48bbdb34c.png
任何矩陣和單位矩陣相乘的結果都仍是原來的矩陣。這就跟標量中的1同樣。
0946767c-8e27-4b04-a31c-e0a97a3fad1b.png遊戲

轉置矩陣

轉置矩陣其實是經過對原矩陣進行轉置運算獲得的。給定一個r×c的矩陣M,它的轉置矩陣能夠表示成MT,這是一個c×r的矩陣。轉置運算就是把原矩陣翻轉一下,即原矩陣的第i行變成了第j列,而第j列變成了第i行。
若是一個矩陣的轉置矩陣是其自己,則咱們稱其爲對稱矩陣遊戲開發

公式

44f3d869-7431-4830-baff-6800e7d85efb.png
對於行矩陣和列矩陣來講,咱們可使用轉置運算來相互轉換:
e013e806-06d2-48b0-8d8c-c9fb7eadeb02.png開發

性質

  • 矩陣轉置的轉置等於原矩陣

6534bf11-41b1-4c80-918c-a5f5eb9c6e9b.png

  • 矩陣的串接的轉置,等於反向串接各個矩陣的轉置。這個性質能夠擴展到更多矩陣相乘的狀況

ae5932c4-0b3e-442d-95df-4586a21f9e61.png

矩陣的行列式

矩陣M的行列式用|M|表示數學

公式

對於2×2矩陣:
659fffd1-922b-4795-b339-2e5f59699091.png
對於3×3矩陣,能夠展開爲2×2矩陣:

$$ A = \left[ \begin{matrix} a & b & c\\ d & e & f \\ g & h & i \end{matrix} \right] $$

a0a2da5d-2549-4082-b1f5-6bfa6b25c303.png
同理,對於4×4矩陣,能夠展開爲3×3矩陣:

$$ A = \left[ \begin{matrix} a & b & c & d\\ e & f & g & h\\ i & j & k & l\\ m & n & o & p \end{matrix} \right] $$

271631ed-fc5d-4a8f-a769-b56102705d79.png
能夠理解爲對於矩陣第一行的每一個元素,都乘以去除該元素所在行和列後剩下矩陣的行列式,而後把結果按照+ -+-的規律加/減起來。

逆矩陣

給定一個方陣M,它的逆矩陣用M-1來表示。逆矩陣最重要的性質就是把M和M-1相乘,那麼它們的結果將會是一個單位矩陣
只有方陣纔有逆矩陣,且並非全部的方陣都有逆矩陣。
如何判斷一個矩陣是否可逆:若是一個矩陣的行列式不爲0,那麼它就是可逆的。
例如全部元素都爲0的方陣就沒有逆矩陣
若是一個矩陣有對應的逆矩陣,咱們就說這個矩陣是可逆的或者說是非奇異的。不然這個矩陣就是不可逆的或者說是奇異的

公式

8077a364-5f87-4443-9322-712c552abed7.png

性質

  • 逆矩陣的逆矩陣是原矩陣自己

eaf59fe6-3e92-471a-9897-3455f417f00d.png

  • 單位矩陣的逆矩陣是它自己

e851b541-8a9a-440e-a867-7be488ae9630.png

  • 轉置矩陣的逆矩陣是逆矩陣的轉置

3a34e126-cc7a-41a1-9544-3f0c30c3d664.png

  • 矩陣串接相乘後的逆矩陣等於反向串接各個矩陣的逆矩陣。這個性質能夠擴展到更多矩陣相乘的狀況

4b4a5a3f-a9bd-4ca0-887f-58dce852e150.png

逆矩陣是有幾何意義的,一個矩陣能夠表示一個變換,而逆矩陣容許咱們還原這個變換。假設,咱們使用變換矩陣M對矢量進行了一次變換,而後再使用M的逆矩陣M-1進行另外一次變換,那麼咱們會獲得原來的矢量。

正交矩陣

若是一個方陣M和它的轉置矩陣的乘積是單位矩陣的話,咱們就稱這個矩陣是正交的
再結合逆矩陣的公式,咱們能夠知道,若是一個矩陣是正交的,那麼它的轉置矩陣和逆矩陣是同樣的

公式

6cca82c6-d906-4adf-9c81-eafe53f95ba3.png
a738cb34-a871-463a-a8dd-0a865d0f7ba0.png
在三維變換中,咱們常常會使用逆矩陣來求解反向的變換。但逆矩陣的求解每每計算量很大,而若是咱們能夠肯定這個矩陣是正交矩陣的話,就能夠直接經過轉置矩陣獲得逆矩陣。
那麼如何判斷的一個矩陣是不是正交矩陣呢,固然能夠經過公式計算判斷,但這仍然須要必定的計算量,有時候咱們更但願不經過計算,而根據一個矩陣的構造過程來判斷這個矩陣是不是正交矩陣
根據正交矩陣的定義能夠獲得:
be98a6c7-3a25-4062-be29-6e2e8a9981d6.png
這樣,咱們就有個9個等式:
27be4ee2-cf91-43ef-b6bc-4cf6f9169412.png
能夠獲得以下結論:

  • 矩陣的每一行(即c1,c2,c3)都是單位矢量,由於它們與本身的點積爲1
  • 矩陣的每一行(即c1,c2,c3)都互相垂直,由於它們互相的點積爲0(參考點積的公式|a||b|cosθ)
  • 上述的兩條結論對每一列也一樣適用。由於M是正交矩陣的話,MT也是正交矩陣

也就說若是一個矩陣知足上面的條件,那麼它就是一個正交矩陣。

行矩陣仍是列矩陣

因爲一個矢量既能夠轉換成一個行矩陣也能夠轉換成列矩陣,雖然它們自己是沒有區別的,但當咱們須要把它和另外一個矩陣相乘時,就會出現差別,由於矩陣的乘法是不知足交換律的。
在Unity中,常規作法是把矢量放在矩陣的右側,即把矢量轉換成列矩陣來進行計算。此時咱們的閱讀順序是從右到左的。即對矢量v先使用A進行變換,再使用B進行變換,最後使用C進行變換。
95eeaa2f-3da3-44d3-a3c7-2891629e9a75.jpg

矩陣的幾何意義:變換

在遊戲的世界中,變換通常包含了旋轉,縮放和平移。遊戲開發人員但願給定一個點或矢量,再給定一個變換,就能夠經過某個數學運算來求得新的點和矢量。而使用矩陣能夠完美地解決這個問題。

什麼是變換

變換指的是咱們把一些數據,如點,方向矢量,甚至是顏色等,經過某種方式進行轉換的過程。
線性變換指的是那些能夠保留矢量加和標量乘的變換。用數學公式來表示這兩個條件就是:
03cdfd38-3890-49b0-a01c-c18aa12a2950.png
縮放就是一種線性變換,例如f(x) = 2x,能夠表示一個大小爲2的統一縮放
旋轉也是一種線性變換
若是咱們要對一個三維的矢量進行線性變換,那麼僅僅使用3×3的矩陣就能夠表示全部的線性變換。
線性變換除了包括旋轉和縮放外,還包括錯切,鏡像(反射),正交投影等。
仿射變換就是合併線性變換和平移變換的變換類型。仿射變換可使用一個4×4的矩陣來表示。

齊次座標

因爲3×3矩陣不能表示一個平移操做,咱們就將其擴展到了4×4的矩陣。爲此,咱們還須要把原來的三維矢量轉換成四維座標,也就是齊次座標(齊次座標的維度能夠超過四維,但本文泛指四維齊次座標)。
如何把一個三維矢量轉換成四維矢量呢:

  • 對於一個點,從三維座標轉換成齊次座標就是把w份量設置爲1
  • 對於方向矢量,須要把w份量設置爲0。這樣當用4×4矩陣進行變換時,平移的效果會被忽略(由於方向矢量沒有位置)。

基礎變換矩陣

咱們把表示純平移,純旋轉和純縮放的變換矩陣叫作基礎變換矩陣
咱們能夠把一個基礎變換矩陣分解成4個組成部分:

$$ \left[ \begin{matrix} M_{3×3} & T_{3×1}\\ 0_{1×3} & 1 \end{matrix} \right] $$

其中,左上角的矩陣M3×3表示旋轉和縮放,T3×1表示平移,01×3是零矩陣,右下角的元素是標量1(第四行是(0 0 0 1)的緣由是保證w份量保持不變)。

平移矩陣

咱們可使用矩陣乘法來表示對一個點進行平移變換,以下所示,從結果很容易看出這個矩陣爲何有平移效果:點的x, y, z份量分別增長了一個位置偏移,即把點(x, y, z)在空間中平移了(tx, ty, tz)個單位。
0a0c6810-97b3-42e4-b8d7-11520e21f594.png
對一個方向矢量進行平移變換,不會對其產生任何影響:
d1701cf2-c007-43bf-815f-6dc6a53c8832.png
平移矩陣的逆矩陣就是反向平移獲得的矩陣。能夠看出平移矩陣並非一個正交矩陣。
4a9170ef-f25b-438d-a2aa-57af324499a7.png

縮放矩陣

咱們可使用矩陣乘法來表示一個縮放變換:
1d562efb-2f98-440f-895b-8adaff503f13.png
對方向矢量一樣能夠進行縮放:
ff3556cd-3fcb-4dfb-82fc-a661378daee0.png
若是縮放係數Kx = Ky = Kz,咱們把這樣的縮放稱爲統一縮放,不然稱爲非統一縮放。從外觀上看,統一縮放是擴大整個模型,而非統一縮放會擠壓或拉伸模型,改變與模型相關的角度和比例。
縮放矩陣的逆矩陣是使用原縮放矩陣係數的倒數來進行縮放,縮放矩陣通常也不是正交矩陣。
969e677b-f4f9-4f9d-9c80-ddf1942edc8b.png

旋轉矩陣

在學習三維空間下的旋轉矩陣前,咱們先來看一下二維空間下的旋轉,向量v旋轉θ度,獲得v':
4347fce3-f880-4b17-8ff3-acdfe8b61f09.png
假設v的模爲r,則旋轉前v的x = r * cosΦy = r * sinΦ
旋轉後v'的x' = r * cos( Φ + θ )y' = r * sin( Φ + θ )
根據三角函數的展開公式:

$$ cos( Φ + θ ) = cosΦ*cosθ - sinΦ*sinθ\\ sin( Φ + θ ) = sinΦcosθ + sinθ*cosΦ $$

可得v'的x' = x*cosθ - y*sinθy' = x*sinθ + y*cosθ,即
b26deaf3-fce8-40c5-9557-48df9910d731.png
讓咱們再回到三維空間,上面的二維旋轉,能夠理解爲在三維中繞z軸的旋轉正方向進行旋轉(讀者能夠想象在左手座標系下,對於上圖的二維旋轉,z軸正方向是指向屏幕內部的,根據左手法則,繞z軸旋轉的正方向就是由x到y)
所以,三維空間下的繞z軸的旋轉矩陣以下所示,由於繞z軸旋轉,因此z座標應該是保持不變的。
ac1f686c-a9c7-4ec4-a3ab-957099178952.png
同理,繞x軸的旋轉以下所示:
ea82f69e-2690-48ba-b4d1-7d98cbe12fdf.png
繞y軸的旋轉以下所示:
1230486e-7740-46b2-8aea-6bc02715ddd0.png

旋轉矩陣是正交矩陣,所以旋轉矩陣的逆矩陣等於其轉置矩陣。並且多個旋轉矩陣之間的串聯一樣是正交的

複合變換

咱們能夠把平移,旋轉和縮放組合起來,造成一個複雜的變換過程。複合變換能夠經過矩陣的串聯來實現。例如:
5d777cca-6ffa-4e8b-9720-d2d746caf51f.png
因爲矩陣乘法不知足交換律,所以矩陣乘法的順序很重要。在大多數狀況下,咱們約定變換的順序是先縮放,再旋轉,最後平移。
除了須要注意不一樣類型的變換順序外,還須要當心旋轉的變換順序。若是咱們須要同時繞着3個軸進行旋轉,那麼應該按什麼樣的旋轉順序呢?
在Unity中,這個旋轉順序是zxy
旋轉時使用的座標系也有如下兩種選擇:

  • 繞座標系E下的z軸旋轉θz,繞座標系E下的x軸旋轉θx,繞座標系E下的y軸旋轉θy
  • 繞座標系E下的z軸旋轉θz後,座標系也繞z軸旋轉θz,新的座標系記作E'。再在E'下繞x軸旋轉θx,座標系也作一樣的旋轉,記作E''。再在E''下,繞y軸旋轉θy。即旋轉的時候,座標系也一塊兒轉動。

上述兩種狀況的結果是不同的,但若是把它們的順序顛倒一下,獲得的結果就會是同樣的。
對於第二種狀況,yxz旋轉順序的變換矩陣是M = Mrotate_z Mrotate_x Mrotate_y,則對於第一種狀況,zxy的旋轉矩陣也是M = Mrotate_z Mrotate_x Mrotate_y。Untiy的zxy旋轉順序指的就是在第一種狀況下旋轉的順序。

相關文章
相關標籤/搜索