(第四章)OpenGL超级宝典学习:必要的数学知识

news2025/1/17 18:09:16
必要的数学知识

前言

在本章当中,作者着重介绍了几个和3D图形学重要的数学知识,线性代数基础好的同学可以直接绕过本章,说实话这篇博客写到这里,我是非常犹豫的,本章节的内容可以说是很基础,但是相当的枯燥,犹豫再三,还是决定一点点写出来吧,好记性不如烂笔头,妈耶


★提高阅读体验★

👉 ♠一级标题 👈

👉 ♥二级标题 👈

👉 ♥ 三级标题 👈

👉 ♥ 四级标题 👈


目录

  • ♠ 向量
    • ♥ 单位向量
    • ♥ 归一化
    • ♥ 点乘
      • ♣ 代数定义
      • ♣ 几何意义
    • ♥ 叉乘
    • ♥ 向量的长度
    • ♥ 反射和折射
  • ♠ 矩阵
    • ♥ 矩阵的构造和运算符
  • ♠ 坐标系
    • ♥ 对象坐标
    • ♥ 世界坐标
    • ♥ 视图坐标
    • ♥ 裁剪和标准化设备空间
  • ♠ 坐标转换
    • ♥ 单位矩阵
      • ♣ 向量和矩阵的乘积
    • ♥ 平移矩阵
    • ♥ 旋转矩阵
      • ♣ 旋转矩阵的推导
      • ♣ 三维旋转矩阵
    • ♥ 欧拉角
    • ♥ 缩放矩阵
  • ♠ 转换
    • ♥ 链接转换
    • ♥ 四元数
    • ♥ 模型——视图转换
      • ♣ 观察矩阵
    • ♥ 投影转换
      • ♣ 透视矩阵
      • ♣ 正交矩阵
  • ♠ 插值、直线、曲线、样条
    • ♥ 插值
    • ♥ 曲线
    • ♥ 样条
  • ♠ 参考文章
  • ♠ 推送
  • ♠ 结语


♠ 向量

指具有大小和方向的量。它可以形象化地表示为带箭头的线段。箭头所指:代表向量的方向;线段长度:代表向量的大小,在OpenGL当中我们用到的多是三维向量

在这里插入图片描述

如上图所示(x,y,z)就是一个标准向量,箭头方向就是向量方向,从原点到xyz点的距离就是向量大小


♥ 单位向量

大小为1的向量,我们统称为单位向量,如:

  • (1,0,0)
  • (0,1,0)
  • (0,0,1)

♥ 归一化

如果一个向量不是单位向量,我们把它变成单位向量,该过程称为归一化,例如

  • (15,0,0) ——> (1,0,0)

归一化的向量只有方向和原来是相同的,大小变成了1


♥ 点乘

传送门:点乘百度百科


♣ 代数定义

两个向量 a ⃗ \vec{a} a , 和 b ⃗ \vec{b} b 的数量积称为点乘(又叫内积、点积),计算方式如下图所示

在这里插入图片描述


♣ 几何意义

两个向量 a ⃗ \vec{a} a , 和 b ⃗ \vec{b} b ,夹角为θ,点乘计算方式如下

在这里插入图片描述

由公式可以看出,如果 a ⃗ \vec{a} a b ⃗ \vec{b} b 都是单位向量(长度1),那么点乘结果就是夹角的余弦值

注:该定义只对二维和三维空间有效

在这里插入图片描述
这个运算可以简单地理解为:在点积运算中,第一个向量投影到第二个向量上(这里,向量的顺序是不重要的,点积运算是可交换的)


♥ 叉乘

向量积,数学中又称外积、叉积,物理中称矢积、叉乘,是一种在向量空间中向量的二元运算。与点积不同,它的运算结果是一个向量而不是一个标量。并且两个向量的叉积与这两个向量垂直

传送门:点乘百度百科

两个向量 a ⃗ \vec{a} a , 和 b ⃗ \vec{b} b ,夹角为θ,叉乘的计算方式如下图所示

在这里插入图片描述

a ⃗ \vec{a} a b ⃗ \vec{b} b 向量的向量积的方向与这两个向量所在平面垂直,且遵守右手定则

在这里插入图片描述


♥ 向量的长度

就是向量的模,根据勾股定理计算向量在多维空间的长度

在这里插入图片描述


♥ 反射和折射

在给定入射向量和表面法向量(垂直于反射面的向量)的情况下,我们可以获得反射向量,并且在给定折射率的情况下,我们可以得出新的折射方向

在这里插入图片描述


♠ 矩阵

传送门:矩阵百度百科

在数学中,矩阵(Matrix)是一个按照长方阵列排列的复数或实数集合, 如下图所示

在这里插入图片描述

我们为什么要用到矩阵呢,在这里就先简单的理解一下即可,即:

矩阵可以很便捷的表示向量在空间中的变化,例如: a ⃗ \vec{a} a 和矩阵B的乘积表示 a ⃗ \vec{a} a 按照矩阵B的规则去变化,矩阵B可能代表旋转,缩放或者移动等复合的多元变化

注:OpenGL编程中我们可使用的矩阵大小普遍是2X2、3X3、4X4


♥ 矩阵的构造和运算符

OpenGL中并非将一个4X4的矩阵作为一个二维数组来运算,而是作为一个包含16个浮点值的单个数组来计算,这16个值可以代表空间中的一个具体位置,以及3条坐标轴相对于观察者的方向

在这里插入图片描述

左上角的三列元素表示为x、y、z轴的方向,相互垂直(正交),第四列表示为xyz的位置或平移距离

注:OpenGL的矩阵为列优先,一列一列的读取,例如上图的读取顺序:A00、A01、A02、A03、A10、A11

在文章底部有一些参考文章,参考文章内写了向量和一些变换矩阵的推导,看一遍就很容易理解了,作者写的很详细,很棒!


♠ 坐标系

OpenGL对顶点的处理过程包含了一系列的转换过程,一系列转换可以表示成一个矩阵,而矩阵的名字往往是根据当前的坐标空间来命名,例如:将一个对象的顶点从模型空间转换到视图空间的矩阵通常称为模型——视图矩阵


♥ 对象坐标

每个模型在建模的时候都有自己的坐标系,它围绕自身坐标系去进行变化,旋转、平移、缩放等

在这里插入图片描述

如上图所示的模型,有以自己为中心的坐标系


♥ 世界坐标

这个就比较好理解吧,形容所有事物的位置都需要一个参考系,参考系的原点就是世界坐标的原点,例如unity或者cocos中的scene就是世界坐标系,scene的原点就是世界坐标原点


♥ 视图坐标

通常也称为摄像机或视点坐标,是以观察者的角度建立的坐标系,以unity场景摄像机为例,参照下图

在这里插入图片描述


♥ 裁剪和标准化设备空间

在上一章管线相关的内容是我们曾经提及到裁剪和标准化设备空间,这里再聊一下,因为设备分标率不同,在显示前OpenGL会把接收到的顶点坐标根据四分量的w做一次分割,最终所有的顶点坐标会变成-1到1大小的值,不满足显示条件的顶点会被裁剪剔除


♠ 坐标转换

我们通过向量和矩阵相乘将坐标从一个空间转移到另一个空间,这一模块我们来看看一些常见的变换矩阵,以及和矩阵相乘的计算方式


♥ 单位矩阵

在这里插入图片描述

上图就是一个4X4的单位矩阵,除对角线是1其他全是0


♣ 向量和矩阵的乘积

在这里插入图片描述

上图是向量(x,y,z,w)和单位矩阵相乘的结果,任意向量和单位矩阵相乘都是他自身


♥ 平移矩阵

我们已经知道了向量和矩阵的乘法公式,我们来看一个简单的平移矩阵

在这里插入图片描述

在默认w是1.0的情况下,上述矩阵代表沿x轴移动1,y轴移动2,z轴移动3


♥ 旋转矩阵

若是要围绕3条坐标轴任意一条去旋转,需要用到旋转矩阵,旋转矩阵稍微麻烦一点,我们可以先从二维坐标系来简单推导一下


♣ 旋转矩阵的推导

注:以下推导逻辑摘自为什么OpenGL里的变换矩阵是4x4的

在这里插入图片描述

我们来简单分析一下坐标系和内容

  • 二维坐标系包含xy轴
  • 2维向量 O A ⃗ \vec{OA} OA 的值是(3,2)
  • 2维向量 O j ⃗ \vec{Oj} Oj O i ⃗ \vec{Oi} Oi 都是长度为1的标准向量

在这里插入图片描述

根据上图我们可以看出, O A ⃗ \vec{OA} OA 向量是和单位矩阵相乘沿x移动3,沿y移动2的结果,而单位矩阵是由 O j ⃗ \vec{Oj} Oj O i ⃗ \vec{Oi} Oi 组成的

在这里插入图片描述

现在我们将坐标系逆时针旋转45°,得到一个新的坐标系,我们继续按图分析

  • O A ⃗ \vec{OA} OA 变成了 O A ′ ⃗ \vec{OA′} OA
  • O j ⃗ \vec{Oj} Oj 变成了 O j ′ ⃗ \vec{Oj′} Oj
  • O i ⃗ \vec{Oi} Oi 变成了 O i ′ ⃗ \vec{Oi′} Oi
  • i′的坐标是(cos45°, sin45°)
  • j′的坐标是(-sin45°, cos45°)

在这里插入图片描述

根据之前的旋转公式新向量 O A ′ ⃗ \vec{OA′} OA 的结果就是上图的结果,因为 O A ′ ⃗ \vec{OA′} OA 相对于 O i ′ ⃗ 和 \vec{Oi′}和 Oi \vec{Oj′}的相对位置是没有变化的,由此我们得到旋转的二维矩阵

在这里插入图片描述


♣ 三维旋转矩阵

我们已经在二维推算出了旋转矩阵,那么三维就是多了一个轴的问题

  • 绕z轴旋转

在这里插入图片描述

看起来是不是很熟悉,我们在xy二维坐标系上的旋转,切换到三维坐标系,其实就是在绕z轴旋转,我们保持z轴不变,就很轻易的得到了绕z轴旋转的矩阵

  • 绕x轴旋转

在这里插入图片描述

  • 绕y轴旋转

在这里插入图片描述

同理我们可以用类似的方式推导出绕x轴和绕y轴旋转的矩阵


♥ 欧拉角

空间中三个角的集合,每个角代表围绕三个正交向量xyz其中一个进行旋转,自身欧拉角旋转会出现万向节死锁,不建议用


♥ 缩放矩阵

在这里插入图片描述

sx、sy、sz分别对应xyz轴的缩放,图示是x轴长度变为原来的两倍


♠ 转换

我们已经了解到了向量从一个空间转换到另一个空间,只需要和相应的矩阵相乘就可以了,我们往往不会用向量一个一个去乘需要的矩阵,而是将需要的矩阵先相乘,最后再和向量乘


♥ 链接转换

vmath::mat4 translation_matrix = vmath::translate(4.0f, 10.0f, -20.0f);
vmath::mat4 rotation_matrix = vmath::rotate(45.0f, vmath::vec3(0.0f, 1.0f, 0.0f));
vmath::mat4 composite_matrix = translation_matrix * rotation_matrix;
vmath::vec4 input_vertex = vmath::vec4(...);

vmath::vec4 transformed_vertex = composite_matrix *
                                 input_vertex;

看上述代码为书中案例,一个移动矩阵translation_matrix和一个旋转矩阵rotation_matrix,我们先将两个矩阵相乘得到复合矩阵composite_matrix,再将复合矩阵和向量相乘

注意: 变换顺序和编码顺序是反的,上述代码先旋转再移动


♥ 四元数

传送门:四元数百度百科

一系列的旋转可以表示为一系列的四元数相乘,生成一个最终的四元数表示所有的旋转。虽然可以制定一串矩阵表示绕着各个笛卡尔轴进行旋转,然后将它们乘在一起,但这种方法容易产生万向节死锁。如果我们用一系列四元组来做同样的事情,将不会发生万向节死锁。为了便于我们编码,vmath库包含vmath::quaternion类实现了这里描述的大部分功能


♥ 模型——视图转换

在一个简单的OpenGL应用中最常用的一个变换是将模型从模型空间转换到视图空间,表达这一变换的矩阵称为模型——视图转换


♣ 观察矩阵

简单理解就是为观察点或者说是摄像机设置一个观察矩阵,目的就是计算模型从世界坐标系转换到观察坐标系后的坐标

在这里插入图片描述

以上图untiy的scene空间和game空间为例,模型cube在scene内是处于世界坐标系内,通过摄像机或者说通过观察矩阵,最终转为在Game视窗内的观察坐标系


♥ 投影转换

投影转换在模型——视图转换后用于顶点,明确了一个完成场景如何投影成屏幕上的最终图像,我们常用的两种投影即:正交投影和透视投影


♣ 透视矩阵

透视投影往往更加真实,和真实的屋里空间相同,近大远小,有透视效果

在这里插入图片描述
在这里插入图片描述

透视矩阵包含几个参数,同样以unity的透视相机为例,其参数包含远近平面的距离和左右上下裁剪平面的世界坐标,整体的形状呈现一个平头椎体


♣ 正交矩阵

正交投影和透视投影不同,在正交投影下的物体无论远近都会按照指定的尺寸绘制在屏幕上,常用于2D游戏

在这里插入图片描述

构建正交矩阵所用参数为视图空间内左右上下场景边界坐标,以及远近平面的位置,以unity正交相机为例,其形状为一个长方体


♠ 插值、直线、曲线、样条

寻找一系列已知点之间的值的过程叫做插值


♥ 插值

在这里插入图片描述

vec4 mix(vec4 A, vec4 B, float t)

这一快没啥好讲的,就是找两个向量中间的某个值,看书中例子点P就是向量A和向量t的和,平滑的改变t可以看到从A平滑的变换到B,GLSL有专门的一个接口处理线性插值


♥ 曲线

在真实世界中,对象以平滑的曲线移动并且平滑地加速、减速。一个曲线可用三个或更多控制点所表示。对大多数曲线来说,有超过三个控制点,其中两个是端点,其他的点定义了曲线的形状

在这里插入图片描述

上图是一个简单的贝塞尔曲线
A和C是曲线的端点,B点的位置可以定义曲线的形状
连接AB、BC,然后我们在两条直线上用简单的线性插值寻找一对新的点:D和E
连接DE,然后沿着它做插值计算寻找一个新的点:P

数学公式:

D = A + t(B - A)
E = B + t(C - B)
P = D + t(E - D)

随着插值t的不断变大,P点会沿着曲线从A移动到C

在这里插入图片描述

拥有两个控制点B和C的贝塞尔曲线


♥ 样条

一个样条实际上就是一个长的曲线,它由几个小的曲线(比如贝塞尔)组成,每个小的曲线都在局部范围内定义它的形状。其中表示各个小曲线端点的控制点至少是被各个线段共享的(就是这些端点将小曲线组成一个样条。这样的控制点被称为焊接点 welds,它们之间的控制点被称为绳结 knots),并且通常内部控制点的一个或多个也在邻近线段间以某种方式共享或联系起来。任意数目的曲线都可以这种方式连接起来,可以形成任意长的路径

在这里插入图片描述

上图便是由10个控制点组成的一个样条,它包含了三个贝塞尔曲线,为了使动作平滑且流畅,对于控制点位置的选择必须非常小心。插值点P的值的变化速率(即P的速度)随着关于t的曲线方程而不同。如果这个函数是非连续的,那P就会突然变向,然后我们的对象就会到处乱跳


♠ 参考文章

OpenGL超级宝典(第7版)笔记16 向量 矩阵 它们都有什么用

为什么OpenGL里的变换矩阵是4x4的?

线性代数基础——矩阵和向量乘法


♠ 推送

  • Github
https://github.com/KingSun5

♠ 结语

本章并没有特别深奥的数学问题,很多用到的东西像矩阵相乘、点乘、叉乘、四元数在各种数学库中均有封装,知道这个东西并且会用就可以了,若是觉得博主的文章写的不错,不妨关注一下博主,点赞一下博文,另博主能力有限,若文中有出现什么错误的地方,欢迎各位评论指摘。
本文属于原创文章,转载请著名作者出处并置顶!!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/155710.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SSM框架01_Spring

有一个效应叫知识诅咒:自己一旦知道了某事,就无法想象这件事在未知者眼中的样子。00-Spring课程介绍01-初识Spring今天所学的Spring其实是Spring家族中的Spring Framework;Spring Fra是Spring家族中其他框架的底层基础,学好Spring可以为其他S…

Morse1题解

原理摩尔斯电码和电报简单说一下电报和摩尔斯电码的原理最简单的电报模型就是一个电源,一个开关和一个电磁铁当需要长距离使用时候,需要用到继电器按下开关,电磁铁会吸引磁铁长按开关,电磁铁就会闭合一段时间,留下一划…

Jenkins集成GitLab Webhooks自动化构建

JenkinsGitLab Webhooks自动构建项目1 构建步骤1.1 Jenkins中设置构建触发器1.2 Build Authorization Token Root插件安装1.3 GitLab配置Webhooks2 测试webhooks2.1 测试推送事件2.2 测试合并请求事件2.3 代码修改提交测试1 构建步骤 1.1 Jenkins中设置构建触发器 这里先随便写…

Markdown与DITA比较

Markdown是一种轻量级标记语言,创始人为John Gruber。它允许人们使用易读易写的纯文本格式编写文档,然后转换成有效的HTML文档。这种语言吸收了很多在电子邮件中已有的纯文本标记的特性。由于Markdown的轻量化、易读易写特性,并且对于图片&am…

第一章Mybatis基础操作学习

文章目录MyBatis简介MyBatis历史MyBatis特性和其它持久化层技术对比搭建MyBatis开发环境创建maven工程创建MyBatis的核心配置文件创建mapper接口创建MyBatis的映射文件通过junit测试功能加入log4j日志功能不带参数的增删改查Mapper接口的编写对应Mapper接口的xml文件编写核心配…

【Python基础】如何使用pycharm

1、设置Python 解释器 在任何项目,第一步就是设置Python 解释器,就是那个Python.exe 在File->Setting->Projec: xxx 下找到 Project Interpreter。然后修改为你需要的 Python 解释器。注意这个地方一定要注意的是:在选择 Python 解释…

Dubbo 学习笔记

Dubbo 学习笔记 1.基础知识 1.1 分布式基础理论 1.1.1 什么是分布式系统? 《分布式系统原理与范型》定义: 分布式系统是若干独立计算机的集合,这些计算机对于用户来说就像单个相关系统分布式系统(distributed system&#xf…

java基于ssm蛋糕店蛋糕商城蛋糕系统网站源码

简介 java使用ssm开发的蛋糕商城系统,用户可以注册浏览商品,加入购物车或者直接下单购买,在个人中心管理收货地址和订单,管理员也就是商家登录后台可以发布商品,上下架商品,处理待发货订单等。 演示视频 …

HTML贪吃蛇游戏源码(穿墙)

演示 完整HTML <!DOCTYPE html> <html> <head><meta charset"utf-8"><title><・)))><<</title><link rel"shortcut icon" href"${ctx}/image/snake_eating.png"><meta name"ref…

中科大2006年复试机试题

中科大2006年复试机试题 文章目录中科大2006年复试机试题第一题问题描述解题思路及代码第二题问题描述解题思路及代码第三题问题描述解题思路及代码第四题问题描述解题思路及代码第五题问题描述解题思路及代码第六题问题描述解题思路及代码第一题 问题描述 求矩阵的转置。 给…

three.js入门篇6之 环境贴图、经纬线映射贴图与高动态范围成像HDR

目录013-1 环境贴图013-2 经纬度映射贴图与HDR013-1 环境贴图 就是把周边的环境&#xff0c;贴在物体的表面之上 注意&#xff1a;px&#xff1a;x轴正向&#xff0c;nx&#xff1a;x轴负向 import * as THREE from "three" // console.log(main.js,THREE);// 导入…

06什么是Fabless?什么是IDM?

Fabless是SIC&#xff08;半导体集成电路&#xff09;行业中无生产线设计公司的简称&#xff0c;只搞设计的无晶圆厂半导体公司&#xff0c;生产交给像台积电这样的代工厂去做。 IDM是整合元件制造商&#xff0c;像英特尔这样既设计又制造的就叫IDM&#xff0c;因为规模大&…

对于字节,16进制,2进制, 0xFF,位移的一些杂记

1.普通字符串95 对应的16进制的展示&#xff0c;使用工具查看如下图 下图为普通字符串 下图为95对应的16进制 95对应的16进制字符串为39 35》39代表一个字节 35代表另一个字节 &#xff08;一个字节是由两位16进制字符串组成&#xff0c;比如39或35&#xff09; 1个字节对应…

select for update加了行锁还是表锁?

最近在开发需求的时候&#xff0c;用到了select......for update。在代码评审的时候&#xff0c;一位同事说 &#xff0c;唯一索引一个非索引字段&#xff0c;是否可能会锁全表呢&#xff1f;本文将通过9个实验操作的例子&#xff0c;给大家验证select......for update到底加了…

迁移环境时,忘记私钥证书密码怎么办?

知行之桥的版本在进行不断更新&#xff0c;相较之前的版本而言&#xff0c;知行之桥每一次更新的版本&#xff0c;无论在操作还是功能亦或是便利性上都有更好的优势&#xff0c;因此不少企业会在新版本更新之后果断选择新的版本&#xff0c;企业选择版本更新之后&#xff0c;需…

He3 新版上新

系统功能更新 支持拖动工具&#xff0c;调整位置 支持置顶 支持自定义分类 新增工具 Paseto 生成器 2. 文本分析 JSON 转 PHP&#xff0c;YAML 转 PHP UTF7 编码、UTF7 解码 6. UTM 生成器 CSS 边框圆角生成器 CSV 类转换工具&#xff0c;目前支持 CSV 与 Markdown、HTML、JS…

什么是无代码ITSM工具

拥有强大 ITSM 团队的企业已经能够生存下来&#xff0c;并且在某些情况下在整个大流行期间表现出色。成功的 IT 团队以其在日常运营中断时快速恢复的能力而闻名。 当您需要重新组织服务交付流程时&#xff0c;ITSM 平台可以减少工程工作量&#xff0c;这对于制定弹性 ITSM 战略…

Python学习笔记——元组

Python将不能修改的值称为不可变的&#xff0c;而不可变的列表被称为元组。定义元组元组创建只需要在括号中添加元素&#xff0c;并使用逗号隔开即可。元组使用小括号 ( )&#xff0c;列表使用方括号 [ ]。定义元组后&#xff0c;就可以使用索引来访问其元素&#xff0c;就像访…

ansible作业二

ansible匹配自定义路径清单文件 查看当前匹配的清单文件路径 [rootserver ~]# ansible --version ansible [core 2.13.5]config file /etc/ansible/ansible.cfg --- 默认配置文件configured module search path [/root/.ansible/plugins/modules, /usr/share/ansible/plugin…

力扣(LeetCode)1150. 检查一个数是否在数组中占绝大多数(C++/Python3)

遍历 直观思考&#xff0c;一次遍历数组&#xff0c;计数 target 。用 target 出现次数和数组长度的一半做比较&#xff0c;即可得到答案。 class Solution { public:bool isMajorityElement(vector<int>& nums, int target) {int cnt 0;for(auto &x:nums)if(…