美文网首页
OpenGL向量/矩阵、压栈/出栈

OpenGL向量/矩阵、压栈/出栈

作者: Maji1 | 来源:发表于2020-07-13 09:59 被阅读0次

我们大家可能对矩阵的乘法有点模糊了,不用担心!我们在这里只需要先明白向量、矩阵的含义,以及在OpenGL中的作用,系统有提供很多API供我么使用。

GLTools库中有一个组件Math3d,其中包含了大量好用的OpenGL一致的3D数学和数据类型。虽然我们不必亲自进行所有的矩阵和向量的操作,但需要知道它们是什么?以及如何运⽤它们。

定义回顾

  • 向量:只有一行或者一列的数组被称作为向量。我们称为长度为1的向量为单位向量

    如果一个向量不是单位向量,而我们把它缩放到长度为1,这个过程叫做标准化,也叫做单位向量华

  • 矩阵:矩阵是指由数字组成的矩形阵列,并写在方括号中间。

OpenGL里的矩阵/向量使用

向量(x,y,z)表示两个重要的值,方向数量


方向:比如向量(1,0,0)。在X方向为+1,而在Y方向和Z方向则为0;
数量:一个向量的数量就是这个向量的长度。
  • match3d库
    match3d库有两个数据类型,能够表示一个三维或四维向量。
    M3DVector3f可以表示一个三维向量(x,y,z)
    M3DVector4f可以表示一个四维向量(x,y,z,w)w是缩放因子。x、y、z通过除以w来进行缩放。
typedef float M3DVector3f[3];
typedef float M3DVector4f[4];
//声明⼀个三分量向量操作:
M3DVector3f vVector;

//类似,声明⼀个四分量的操作:
M3DVector4f vVectro= {0.0f,0.0f,1.0f,1.0f};

//声明一个三分量顶点数组,例如⽣成⼀个三⻆形
M3DVector3f vVector[] = {
    0.5, 0.0, 0.0,
    0.0, 0.5, 0.0,
    0.0, 0.0, 0.0,    
};

向量可以进行加法、减法计算,但是在开发中比较常用的计算方式是 点乘(dot product叉乘(cross product

向量 点乘

注意:点乘只能发生在两个向量之间。
float m3dDotProduct3(const M3DVector3f u,const M3DVector3f v);
返回的是[-1,1]之见的值,代表这两个向量的余弦值。

float m3dGetAngleBetweenVector3(const M3DVector3f u,const M3DVector3f v);
返回两个向量之间夹角的 弧度值。

2个 单位向量(三维向量)之间进行点乘得到的是一个标量,它表示两个向量之间的夹角。

单位量计算如图:


使用非零向量a 除以它的模b(向量的长度), 就可以得到方向相同的单位向量c
向量 叉乘

2个向量之间叉乘就可以得到另外⼀个向量,新的向量会与原来2个向量定义的平面垂直。 进行叉乘,不必为单位向量;

void m3dCrossProduct3(M3DVector3f result,const M3DVector3f u ,const M3DVector3f v);

矩阵

在其他编程标准中,许多矩阵库定义一个矩阵时,使用二维数组。
OpenGL的约定里,更多倾向使⽤一维数组,这样做的原因是 OpenGL 使用的是 Column-Major(以列为主) 矩阵排序的约定。

  • OpenGL 下的矩阵
typedef float M3DMatrix33f[9];// 3x3矩阵
typedef float M3DMatrix44f[16];// 4x4矩阵

这 16 个值表示空间中⼀个特定的位置,这4列中,每⼀列都是有4个元素组成的向量。
如果将⼀个对象所有的顶点向量乘以这个矩阵,就能让整个对象变换到空间中给定的位置和方向。

  • 单元矩阵初始化方式
// 方式一:
GLFoat m[] = {
  1, 0, 0, 0,      // X 列
  0, 1, 0, 0,      // Y 列
  0, 0, 1, 0,      // Z 列
  0, 0, 0, 1,      // Translation
}
// 方式二:
M3DMatrix44f m = {
  1, 0, 0, 0,      // X 列
  0, 1, 0, 0,      // Y 列
  0, 0, 1, 0,      // Z 列
  0, 0, 0, 1,      // Translation
}
// 方式三:
void m3dLoadIdentity44f(M3DMatrix44f m);

将一个向量乘以 单元矩阵 等于向量本身。

⚠️注意:A * B的前提条件是矩阵 A 的列数 == 矩阵B的行数
⚠️注意:A * B != B * A

  • 在线性代数中为了方便书写,都是从左往右的顺序进行计算的。
    顶点 x 模型矩阵 x 观察矩阵 x 投影矩阵 = 变换后顶点向量
  • 但是在OpenGL中的计算方式是 左乘
    投影矩阵 x 观察矩阵 x 模型矩阵 x 顶点 = 变换后顶点向量
    例如:
    1、modelViewMatrix.MultMatrix(mProjection);//投影矩阵
    2、modelViewMatrix.MultMatrix(mCamera);//观察矩阵
    3、modelViewMatrix.MultMatrix(mObject);//模型矩阵
    我们简单看一下OpenGL内部源码:
// OpenGL 源码
inline void MultMatrix(const M3DMatrix44f mMatrix) {
    M3DMatrix44f mTemp;
    m3dCopyMatrix44(mTemp, pStack[stackPointer]);
    m3dMatrixMultiply44(pStack[stackPointer], mTemp, mMatrix);
}

我们只看第3步,进入OpenGL源码,看到了 矩阵mObjectmodelViewMatrix 的栈顶矩阵pStack[stackPointer]
计算方式是:pStack[stackPointer] = mObject * pStack[stackPointer]
⚠️注意:A * B != B * A , 所以此处的 计算方式 不能理解为 pStack[stackPointer] * mObject
最后将结果赋值给 pStack[stackPointer]覆盖掉原来的值。


专业变换名词

  • 视图变换:指定观察者位置。
    应用到场景中的第一种变换,默认情况下,透视投影中位于原点(0,0,0),并沿着 z 轴负方向进⾏观察 (向显示器内部“看过去”)。
    视图变换将观察者放在你希望的任何位置。并允许在任何⽅向上观察场景,确定视图变换就像在场景中放置观察者并让它指向某一个方向。
    从⼤局上考虑,在应⽤任何其他模型变换之前, 必须先应⽤视图变换。这样做是因为,对于视觉坐标系⽽言,视图变换移动了当前的工作的坐标系,后续的变化都会基于新调整的坐标系进⾏。

  • 模型变换:在场景中移动物体。(物体的平移、旋转、缩放)

//平移
inline void m3dTranslationMatrix44(M3DMatrix44f m, float x, float y, float z)
{ m3dLoadIdentity44(m); m[12] = x; m[13] = y; m[14] = z; }

//旋转
void m3dRotationMatrix44(M3DMatrix44f m, float angle, float x, float y, float z);

//缩放
inline void m3dScaleMatrix44(M3DMatrix44f m, float xScale, float yScale, float zScale)
    { m3dLoadIdentity44(m); m[0] = xScale; m[5] = yScale; m[10] = zScale; }
  • 模型视图变换:描述 视图/模型 变换的二元性。(包含了模型变换和视图变换)
  • 投影:改变视景体大小和设置他的投影方式。
  • 视口:伪变化,对窗口上最终输出进行缩放。

矩阵堆栈的使用

// GLMatrixStack类 这个类的构造函数允许指定堆栈的最大深度,默认的堆栈深度为64.
// 同时这个矩阵堆栈在初始化时,已经在堆栈中包含了单位矩阵。
GLMatrixStack::GLMatrixStack(int iStackDepth = 64);

// 在堆栈顶部载入一个单元矩阵
void GLMatrixStack::LoadIdentity(void);

// 在堆栈顶部载入任何矩阵
void GLMatrixStack::LoadMatrix(const M3DMatrix44f m);

// 矩阵乘以矩阵堆栈顶部矩阵,相乘结果存储到堆栈的顶部
void GLMatrixStack::MultMatrix(const M3DMatrix44f);

// 获取矩阵堆栈顶部的值 GetMatrix 函数
// 为了适应GLShaderManager的使用,或者是获取顶部矩阵的副本
const M3DMatrix44f & GLMatrixStack::GetMatrix(void);
void GLMatrixStack::GetMatrix(M3Datrix44f mMatrix);

压栈/出栈

压栈/出栈
  • 压栈modelViewMatrix.PushMatrix();:

这句代码的意思是压栈,如果 PushMatix() 括号里是空的,就代表是把栈顶的矩阵复制一份,再压栈到它的顶部。如果不是空的,比如是括号里是单元矩阵,那么就代表压入一个单元矩阵到栈顶了。

  • 出栈modelViewMatrix.PopMatrix();:

把栈顶的矩阵出栈,恢复为原始的矩阵堆栈,这样就不会影响后续的操作了。

仿射变换

我们都知道,想要从不同的角度观察一个3D物体,我们有两种方式,一种是移动物体,还有一种就是我们自己移动。在OpenGL中移动观察者就是 仿射变换

相关代码:

//Rotate 函数angle参数是传递的度数,⽽而不不是弧度
void MatrixStack::Rotate(GLfloat angle,GLfloat x,GLfloat y,GLfloat z);
void MatrixStack::Translate(GLfloat x,GLfloat y,GLfloat z);
void MatrixStack::Scale(GLfloat x,GLfloat y,GLfloat z);

使用照相机和角色帧进行移动:GLFrame

//将堆栈的顶部压入任何矩阵
void GLMatrixStack::LoadMatrix(GLFrame &frame);
//矩阵乘以矩阵堆栈顶部的矩阵。相乘结果存储在堆栈的顶部 
void GLMatrixStack::MultMatrix(GLFrame &frame);
//将当前的矩阵压栈
void GLMatrixStack::PushMatrix(GLFrame &frame)

获取照相机矩阵

//GLFrame函数,这个函数用来检索条件适合的观察者矩阵
void GetCameraMatrix(M3DMatrix44f m,bool bRotationOnly = flase);

相关文章

  • OpenGL笔记六:纹理常用API(一)

    前言 期待您移步上篇:OpenGL笔记五:综合实例理解-压栈、出栈、堆栈矩阵相乘、矩阵相乘、向量相乘 纹理 纹理只...

  • OpenGL 出栈压栈理解分析

    1.OpenGL压栈出栈作用概念 1.压栈出栈操作的是矩阵,一般分为模型视图矩阵和投影矩阵 2.出栈压栈是针对顶点...

  • OpenGL-矩阵与向量

    OpenGL的矩阵操作 旋转 平移 缩放 压栈/出栈 单位向量-X轴-(1,0,0) 单位矩阵-X,Y,Z 三轴...

  • OpenGL向量/矩阵、压栈/出栈

    我们大家可能对矩阵的乘法有点模糊了,不用担心!我们在这里只需要先明白向量、矩阵的含义,以及在OpenGL中的作用,...

  • OpenGL_矩阵压栈和出栈

    1. 压栈和出栈的理解 压栈出栈操作的是矩阵 用来记录矩阵的状态 压栈PushMatrix和出栈PopMatrix...

  • OpenGL学习笔记五

    使⽤矩阵堆栈 压栈.出栈 压栈: 存储一个状态出栈: 恢复⼀个状态 仿射变换

  • OpenGL的矩阵压栈与出栈

    初学时,我们通常会对矩阵压栈和出栈这对“难兄难弟”感到疑惑,为什么它们是成对出现的?它们做的又是什么操作呢?那,今...

  • OpenGL-矩阵压栈出栈浅析

    OpenGL绘制图形需要经过变化才能达到用户目的,而变换是通过矩阵进行操作的。 OpenGL一般通过视图变换、模型...

  • OPenGL 中的矩阵压栈/出栈

    我们知道OPenGL 中的图形变化是用矩阵来记录保存的.OpenGL 矩阵基础变化 简单来讲,当你做了一些移动或旋...

  • OpenGL中矩阵堆栈的压栈和出栈操作理解

    OpenGL中矩阵堆栈的频繁压栈和出栈操作往往是入门时最大的门槛,也是最容易造成困惑的地方,今天我们来详细理解一下...

网友评论

      本文标题:OpenGL向量/矩阵、压栈/出栈

      本文链接:https://www.haomeiwen.com/subject/cpqscktx.html