UnityShader数学基础篇
Mathf
Mathf和Math
1、Math是C#中封装好的用于数学计算的工具类,位于System命名空间中。
2、Mathf是Unity中封装好的用于数学计算的工具结构体,位于UnityEngine命名空间中。
Mathf中的常用方法
1.π - PI
print(Mathf.PI);
2.取绝对值 - Abs
print(Mathf.Abs(-10.5f));//10.5
print(Mathf.Abs(-86));//86
3.向上取整 - CeilToInt
print(Mathf.CeilToInt(1.001f));//2
print(Mathf.CeilToInt(5.6f));//6
4.向下取整 - FloorToInt
print(Mathf.FloorToInt(2.999f));//2
print(Mathf.FloorToInt(1.04f));//1
5.钳制函数 - Clamp (传入的数据 ,数据传出最小值 ,数据传出最大值)
int num = 18;
print(Mathf.Clamp(num, 13, 32));//13
print(Mathf.Clamp(num, 13, 32));//32
print(Mathf.Clamp(num, 13, 32));//18
6.获取最大值 - Max
int[] ints = new int[5] {5,9,78,65,23};
print(Mathf.Max(1, 5, 6, 8, 9, 45));//45
print(Mathf.Max(ints));//78
7.获取最小值 - Min
int[] ints2 = new int[5] {-1,5,86,411,20};
print(Mathf.Min(ints2));//-1
print(Mathf.Min(1.2f,5,65,0.86f));//0.86
8.一个数的n次幂 - Pow
print("2的6次方" + Mathf.Pow(2, 6));//64
print("3的4次方" + Mathf.Pow(3, 4));//81
9.四舍五入 - RoundToInt
print("四舍五入" + Mathf.RoundToInt(4.6f));//5
print("四舍五入" + Mathf.RoundToInt(4.3f));//4
10.返回一个数的平方根 - Sqrt
print("平方根" + Mathf.Sqrt(4));//2
print("平方根" + Mathf.Sqrt(9));//3
11.判断一个数是否是2的n次方 - IsPowerOfTwo
print("11:" + Mathf.IsPowerOfTwo(8));//ture
print("11:" + Mathf.IsPowerOfTwo(9));//false
12.判断正负数 - Sign
print("判断正负数" + Mathf.Sign(1));//1
print("判断正负数" + Mathf.Sign(-2));//-1
13.插值运算 - Lerp
1、Lerp函数公式:result = Mathf.Lerp(start, end, t);
2、t为插值系数,取值范围为0~1:result = start + (end - start)*t
float start = 0;
float result = 0;
float time = 0;
void Update()
{
//插值运算用法一
//每帧改变start的值一变化速度先快后慢,位置无限接近,但是不会得到end位置
start = Mathf.Lerp(start,10,Time.deltaTime);
//插值运算用法二
//每帧改变t的值一变化速度匀速,位置每帧result接近end,当t>=1时,得到结果
time += Time.deltaTime;
result = Mathf.Lerp(start,10,time);
}
三角函数
Unity中都是弧度值,让物体曲线移动。
弧度(radian)、角度相互转化
1、1 rad = (180/π)°=> 1 rad = (180/3.14)°= 57.3°
2、弧度转角度:弧度 * 57.3 = 对应的角度
3、Mathf.Rad2Deg
float rad = 1;
float anger = rad * Mathf.Rad2Deg;
print(anger);//57.3
1、1°= (π/180)rad => 1°= (3.14/180)rad = 0.01745 rad
2、角度转弧度:角度 * 0.01745 = 对应的弧度
3、Mathf.Deg2Rad
anger = 1;
rad = anger * Mathf.Deg2Rad;
print(rad);//0.01745
三角函数
注意:Mathf中的三角函数相关函数,传入的参数需要弧度值
Sin() Cos()
print(Mathf.Sin(30 * Mathf.Deg2Rad));//1/2
print(Mathf.Cos(30 * Mathf.Deg2Rad));//sqrt3/2
print(Mathf.Sin(30f));//1/2
反三角函数
注意:反三角函数得到的结果是正弦或者余弦值对应的弧度值
Asin() Acos()
print(Mathf.Asin(0.5f) * Mathf.Rad2Deg);//30
print(Mathf.Acos(0.5f) * Mathf.Rad2Deg);//60
Unity坐标系
世界坐标系
transform.position
transform.rotation
transform.eulerAngles
transform.lossyScale
物体坐标系
1、相对父对象的物体坐标系的位置 本地坐标 相对坐标
2、修改他们会是相对父对象物体坐标系的变化
transform.localPosition;
transform.localRotation;
transform.localEulerAngles;
transform.localScale;
屏幕坐标系
Input.mousePosition;
Screen.width;
Screen.height;
坐标转换
//世界转本地
transform.InverseTransformDirection(Direction); //不受缩放影响(向量)
transform.InverseTransformVector(Vector); //受缩放影响
transform.InverseTransformPoint(pos);
//本地转世界
transform.TransformDirection(localDirection);
transform.TransformVector(localVector);
transform.TransformPoint(localPos);
//世界转屏幕
Camera.main.WorldToScreenPoint(pos);
//屏幕转世界
Camera.main.ScreenToWorldPoint(ScreenPos);
//世界转视口
Camera.main.WorldToViewportPoint(pos);
//视口转世界
Camera.main.ViewportToWorldPoint(ViewportPos);
//视口转屏幕
Camera.main.ViewportToScreenPoint(ViewportPos);
//屏幕转视口
Camera.main.ScreenToViewportPoint(ScreenPos);
向量
1、Vector3这边变量 可以表示一个点 也可以表示一个向量 具体表示什么 是根据我们的具体需求和逻辑决定。
2、如何在Unity里面得到向量,终点减起点,就可以得到向量。点C也可以代表向量,代表的就是oc向量,o是坐标系原点。
3、得到了向量就可以利用vector3中提供的成员属性,得到模长和单位向量。
4、模长相当于可以得到两点之间的距离,单位向量主要是用来进行移动计算的它不会影响我们想要的移动效果。
Vector3 A = new Vector3(1, 2, 3);
Vector3 B = new Vector3(5, 4, 7);
//两点向量
Vector3 AB = B - A;
Vector3 BA = A - B;
//两个物体之间的向量
Vector3 Vec = Object .position - transform.position;
//magnitude 向量的模长
print(Vec.magnitude);
print(Vector3.Distance(Object.position, transform.position));
//单位向量 normalized
print(Vec.normalized);
print(Vec / Vec.magnitude);
向量加减乘除运算
#region 知识点一 向量加法
transform.position += new Vector3(1, 2, 3);
#endregion
#region 知识点二 向量减法
transform.position -= new Vector3(1,2,3);
#endregion
#region 知识点三 向量乘除标量 放大缩小n倍
transform.localScale *= 2;
transform.localScale /= 2;
#endregion
向量点乘(dot product)
公式一:A•B = (a1,b1,c1)•(a2,b2,c2) = a1a2 + b1b2 +c1c3
公式二:A•B = |A||B|cosβ
几何意义:投影,判断前后
求角度:
#region 知识点一 通过点乘判断对象的方位
//Vector3 提供了计算点乘的方法
Debug.DrawRay(transform.position, transform.forward, Color.blue);
Debug.DrawRay(transform.position, Target.position - transform.position, Color.green);
if (Vector3.Dot(transform.forward, Target.position - transform.position) >= 0)
{
print("目标在前方");
}
else
print("目标在后方");
#endregion
#region 知识点二 通过点乘推导公式算出夹角
//公式: 角度 = Acos(单位向量 • 单位向量)
//1、用单位向量算出点乘结果
float DotResult = Vector3.Dot(transform.forward, (Target.position - transform.position).normalized);
//2、用反三角函数得出弧度,然后转为角度
print("角度:" + Mathf.Acos(DotResult) * Mathf.Rad2Deg);
//Vector3中提供了 得到两个向量之间夹角的方法
print("角度:" + Vector3.Angle(transform.forward, Target.position - transform.position));
//作用
//怪物范围检测 角度范围内检测
#endregion
向量叉乘(cross product)
公式:
模计算:|a×b|=|a||b|sinθ,平行四边形面积计算
几何意义:判断左右、法向量,三角形面片朝向
#region 知识点一 叉乘计算
print(Vector3.Cross(A.position, B.position).normalized);
#endregion
if (Vector3.Cross(A.position, B.position).y > 0)
print("B在A的左边");
else
print("B在A的右边");
矩阵乘法
矩阵概念
矩阵的结构是由 m x n 个标量组成。
在程序中,我们用于存储矩阵结构的容器类型有很多选择,最常见的的为:
1、数组(一维、二维都可以)
2、嵌套列表(两个List嵌套)
3、开发工具提供的类或结构体(Unity中的Matrix4x4、Matrix3x2结构体)
矩阵和标量的乘法
矩阵(M)中的每一个标量和标量(k)相乘即可
矩阵和矩阵的乘法
1、首先需要判断两个矩阵是否能够相乘
判断条件:左列右行要相等
2、A和B两个矩阵,AB两个矩阵相乘的结果是C矩阵。
那么C(11) = A(1n).B(n1)、C(12) = A(1n).B(n2)、C(13) = A(1n).B(n3)
解读:C矩阵中的第一行第一列的值等于A中第一行点乘B中第一列。
矩阵之间的乘法
1、不满足交换律
AB ≠ BA
2、满足结合律
(AB)C = A(BC)
ABCDE = (AB)(CD)E = A((BC)D)E
特殊矩阵
- 方块矩阵 —— 行列数相等的矩阵。
- 对角矩阵 —— 只有主对角线有值,其余元素全为零的方阵。
- 单位矩阵 —— 主对角线上的元素均为1 的对角矩阵。
- 数量矩阵 —— 主对角线上的元素为同一值的对角矩阵。
- 转置矩阵 —— 将原始矩阵的行和列互换得到的新矩阵。
- 矩阵转置的转置等于原矩阵 (MT)T = M
- 矩阵串接的转置,等于反向串接各个矩阵的转置 (AB)T =BTAT
逆矩阵
- 逆矩阵必须是一个方阵,并且不是所有矩阵都有逆矩阵。
- 假设一个方阵 M ,它的逆矩阵用 M-1 表示。
- 那么存在 MM-1 = M-1M = E(单位矩阵)
- 如果一个矩阵存在对应的逆矩阵,我们就说该矩阵是可逆的(或称非奇异的)。
- 如果不存在,那么该矩阵为不可逆的(或称奇异的)。
- 判断方式:行列式不为0,那么可逆。
行列式的计算方式
假设矩阵为M,|M| 表示M矩阵的行列式,行列式是一个标量(数值)
计算方法:
1、左下左上画对角,线上数值都相乘,数值数量为行列,数量不够对岸取
2、左下分组加,左上分组减
代数余子式矩阵
标准伴随矩阵
标准伴随矩阵为原矩阵的代数余子式矩阵的转置矩阵。
逆矩阵的计算
1、逆矩阵 = 标准伴随矩阵 / 行列式
M-1 = CT / |M|
2、初等变换
(A E)->(E A-1)
逆矩阵的重要性质
1.逆矩阵的逆矩阵是原矩阵本身 (M-1)-1 = M
2.矩阵乘以自己的逆矩阵等于单位矩阵 MM-1 = M-1M = E
3.单位矩阵的逆矩阵是它本身 E-1 = E
4.转置矩阵的逆矩阵是逆矩阵的转置 (MT)-1 = (M-1)T
5.矩阵串接相乘后的逆矩阵 等于 反向串接各个矩阵的逆矩阵 相乘 (AB)-1 = B-1A-1
6.逆矩阵可以计算矩阵变换的反向变换(M为矩阵,v为一个矢量)
M-1(Mv) = (M-1M)v = Ev = v
正交矩阵
正交矩阵是一种特殊的方阵,正交的意思是垂直
它的特点是:
1、一个方阵和它的转置矩阵相乘为单位矩阵,那么它就是正交矩阵
MMT = MTM = E
2、通过正交矩阵的这一性质,再根据上节课学习的逆矩阵的一个重要性质
MM-1 = M-1M = E
3、我们可以推导出:如果一个矩阵是正交的,那么它的逆矩阵等于其转置矩阵
MT = M-1
4、如果一个矩阵是正交矩阵,那么它的转置矩阵也是正交矩阵
判断是否为正交矩阵
根据正交矩阵的基本概念,我们可以总结出判断一个矩阵是否是正交矩阵的方式有:
- 判断MMT = MTM = E ,满足则为正交矩阵
- 判断矩阵的每一行(列)是否是单位向量
- 判断矩阵的行(列)向量是否彼此正交(垂直)
行列矩阵
一、列矩阵和行矩阵的基本概念
1、列矩阵就是只有一列的矩阵;行矩阵就是只有一行的矩阵。他们一般用于表示向量
2、把向量作为列矩阵和行矩阵与矩阵进行乘法运算时,计算顺序(列在后,行在前)和结果是不同的
二、列矩阵和行矩阵在Unity中的使用规则
1、在Unity的Shader开发中,我们采用列矩阵的形式进行向量计算,利用结合律,我们可以从右往左阅读
CBAv = C(B(Av))
2、如果想要使用行矩阵计算出和列矩阵相同的结果,我们可以乘以变换矩阵的转置矩阵
vATBTCT = (((vAT)BT)CT)
矩阵的几何意义
点和向量能在图像中画出来,那么矩阵可以吗?
矩阵的可视化结果就是:变换
在游戏开发中,如果你看到了一个矩阵,那么基本上你可以认为你看到的是一个变换,这些变换一般包含:平移、旋转、缩放。
比如:我们想要将一个点、一个向量进行一种变换(平移、旋转、缩放)。那么我们可以利用矩阵来进行数学计算,从而达到变换的目的。
我们可以利用矩阵相关知识做什么?
对三维空间中的向量进行平移、旋转、缩放、坐标变换、投影等等计算,这样我们就可以对Shader中的数据进行处理,让其最终在屏幕上的效果是按照我们的需求来呈现的。
什么是变换?
线性变换:指可以保留矢量加和标量乘的变换。缩放、旋转、错切、镜像、正交投影等。
仿射变换:指合并线性变换和平移变换的变换类型。齐次坐标。
持续更新中
UnityShader数学基础篇的更多相关文章
- 【数学基础篇】---详解极限与微分学与Jensen 不等式
一.前述 数学基础知识对机器学习还有深度学习的知识点理解尤为重要,本节主要讲解极限等相关知识. 二.极限 1.例子 当 x 趋于 0 的时候,sin(x) 与 tan(x) 都趋于 0. 但是哪一个趋 ...
- 《神经网络的梯度推导与代码验证》之FNN(DNN)的前向传播和反向推导
在<神经网络的梯度推导与代码验证>之数学基础篇:矩阵微分与求导中,我们总结了一些用于推导神经网络反向梯度求导的重要的数学技巧.此外,通过一个简单的demo,我们初步了解了使用矩阵求导来批量 ...
- 《神经网络的梯度推导与代码验证》之CNN的前向传播和反向梯度推导
在FNN(DNN)的前向传播,反向梯度推导以及代码验证中,我们不仅总结了FNN(DNN)这种神经网络结构的前向传播和反向梯度求导公式,还通过tensorflow的自动求微分工具验证了其准确性.在本篇章 ...
- 《神经网络的梯度推导与代码验证》之vanilla RNN的前向传播和反向梯度推导
在本篇章,我们将专门针对vanilla RNN,也就是所谓的原始RNN这种网络结构进行前向传播介绍和反向梯度推导.更多相关内容请见<神经网络的梯度推导与代码验证>系列介绍. 注意: 本系列 ...
- 《神经网络的梯度推导与代码验证》之LSTM的前向传播和反向梯度推导
前言 在本篇章,我们将专门针对LSTM这种网络结构进行前向传播介绍和反向梯度推导. 关于LSTM的梯度推导,这一块确实挺不好掌握,原因有: 一些经典的deep learning 教程,例如花书缺乏相关 ...
- TGL站长关于常见问题的回复
问题地址: http://www.thegrouplet.com/thread-112923-1-1.html 问题: 网站配有太多的模板是否影响网站加载速度 月光答复: wp不需要删除其他的模板,不 ...
- 视觉SLAM中的数学基础 第二篇 四元数
视觉SLAM中的数学基础 第二篇 四元数 什么是四元数 相比欧拉角,四元数(Quaternion)则是一种紧凑.易于迭代.又不会出现奇异值的表示方法.它在程序中广为使用,例如ROS和几个著名的SLAM ...
- 视觉SLAM中的数学基础 第三篇 李群与李代数
视觉SLAM中的数学基础 第三篇 李群与李代数 前言 在SLAM中,除了表达3D旋转与位移之外,我们还要对它们进行估计,因为SLAM整个过程就是在不断地估计机器人的位姿与地图.为了做这件事,需要对变换 ...
- 视觉SLAM的数学基础 第一篇 3D空间的位置表示
视觉SLAM中的数学基础 第一篇 3D空间的位置表示 前言 转眼间一个学期又将过去,距离我上次写<一起做RGBD SLAM>已经半年之久.<一起做>系列反响很不错,主要由于它为 ...
- 干货来袭!3天0基础Python实战项目快速学会人工智能必学数学基础全套(含源码)(第3天)概率分析篇:条件概率、全概率与贝叶斯公式
第1天:线性代数篇:矩阵.向量.实战编程 第2天:微积分篇:极限与导数.梯度下降.积分.实战编程 第3天:概率分析篇:条件概率与全概率.贝叶斯公式.实战项目 目录 前言 一.概率与机器学习 1.1 概 ...
随机推荐
- #期望dp#洛谷 6835 [Cnoi2020]线形生物
题目 分析 设\(f[i]\)表示由点\(i\)走到点\(i+1\)的期望步数, \(dp[i]\)表示由点1走到点\(i+1\)的期望步数, 那么\(dp\)为\(f\)的前缀和,最后答案为\(dp ...
- openGauss单机部署
openGauss 单机部署 一.安装环境 操作系统:虚拟机 VMware.CentOS7.9 环境设置: (1)虚拟机内存 3G.磁盘 100G (2)系统版本修改 一开始使用了 centos8,无 ...
- 重新整理.net core 计1400篇[三] (.net core 如何源代码调试和查看源码 )
前言 本来这里是写源代码的,因为vs没有那么容易调试查看到源代码,所以先把调试源代码和查看源代码弄完. 正文 需要修改一些vs配置,这个vs也是有要求的,要vs2017. 那么你还需要加载远程符号. ...
- native react 代码智能提示
背景 在vscode 中,虽然有插件可以达到代码提示的效果但是不是很嗨. 所以加上这些: 全局安装typings: npm install typings -g 1 安装react和react-nat ...
- 配置服务器SSH PAM防止暴力破解
正常机器如果开启ssh可能会被暴力破解用户密码,可以配置ssh的pam限制登录失败时可以锁定用户一段时间,就算密码输入正确也无法登录,等恢复之后才可以再次进入,对于不同的发行版配置可能略有不同,对于r ...
- 如何基于香橙派AIpro对视频/图像数据进行预处理
本文分享自华为云社区<如何基于香橙派AIpro对视频/图像数据进行预处理>,作者: 昇腾CANN. 受网络结构和训练方式等因素的影响,绝大多数神经网络模型对输入数据都有格式上的限制.在计算 ...
- ORA-01555:snapshot too old: rollback segment number X with name "XXXX" too small
ORA-01555:snapshot too old: rollback segment number X with name "XXXX" too small 在查询快照的时候 ...
- 阿里云交互式分析与Presto对比分析及使用注意事项
阿里云交互式分析与Presto对比分析及使用注意事项本文由阿里巴巴耿江涛带来以"阿里云交互式分析与Presto对比分析及使用注意事项"为题的演讲.文章首先介绍了Presto以及它的 ...
- 最佳实践:使用阿里云CDN加速OSS访问
简介: 用户直接访问OSS资源,访问速度会受到OSS的下行带宽以及Bucket地域的限制.如果通过CDN来访问OSS资源,带宽上限更高,并且可以将OSS的资源缓存至就近的CDN节点,通过CDN节点进行 ...
- 基于 KubeVela 的机器学习实践
简介:本文主要介绍如何使用 KubeVela 的 AI 插件,来帮助工程师更便捷地完成模型训练及模型服务. 作者:KubeVela 社区 在机器学习浪潮迸发的当下,AI 工程师除了需要训练.调试自己 ...