欧拉角是欧拉在17世纪发明引进的一个数学工具,在三维欧几里得空间内,欧拉角可以确定一个物体的朝向。在解决静态问题上,欧拉角是一个比较完美的解决方案,但在动态问题上,欧拉角有一个万向锁的瑕疵,数学界在后来发明了四元数也叫欧拉参数已经解决了这一问题。Unity也已经很好的规避解决了万向锁的问题,但是在手动旋转物体时还是有可能会遇上这一现象。

在Unity Editor内点击场景中的任意有transform的gameobject,在Inspector面板内可以查看它的transform的position,rotation和scale。

(图1:一个plane物体的Position,Rotation,Scale)

这里的Rotation格式是欧拉角,在原点(0,0,0),手动调整transform的欧拉角xyz值有点像围绕着它的xyz轴旋转。


(图2:修改plane物体的Rotation.x)

其实调整一个物体在inspector内的欧拉角的xyz值并不代表它围绕着某个轴旋转多少度。这个值描述的是该物体当前在静态状态下的欧拉角度。由于“一个物体分别围绕xyz轴转动了多少度”要比“一个物体的当前欧拉角是多少度”更直观好理解,所以我们倾向于按前者来思考。

比如我说一个物体在原点(0,0,0)绕x轴转90度,基本所有人都能想到旋转后的状态是(90,0,0),但如果我说一个物体的欧拉角是(90,-90,-90),很多人都想不到它与前者是同样的角度状态。

但如果用前者,“一个物体分别围绕xyz轴转动多少度”这种动态的思路来理解使用欧拉角,就会出现一个问题:在某些情况下无法通过单一一个角度值的线性变化去表现一个物体不断的旋转。例如当plane物体的欧拉角x=90时,调整y或z的值,plane的旋转表现都是围绕着y轴,也既是我们无法通过调整它的欧拉角属性来让它围绕z轴旋转了。

(图3:当plane的欧拉角x=90时,修改y和z的值,它都是在围绕y轴旋转)

这一特殊情况通常被称为万向锁,Gimbal Lock。我觉得万向锁这词儿有点儿夸张,在这里更通俗直观的说法应该是“以线性的方式修改欧拉角的xyz其中之一的值无法随时随地很好的很方便的旋转一个物体”。

首先我们需要先了解虽然inspector面板上的欧拉角前面显示的是rotation,但在Unity中并不存在欧拉角格式的rotation。在Unity中,一个transform的rotation是Quaterion四元数格式,虽然这个类class也有一个欧拉角变量属性,但当你修改这个欧拉角变量的时候,Unity会自动将它转化为四元数再赋值给四元数的xyzw,而当你读这个值的时候,Unity会再次将四元数rotation转为欧拉角。

Unity无法完全抛弃欧拉角的原因之一是四元数Quaterion的xyzw的值代表的意思非常不直观,例如Quaterion.x的意义为下:
//注:Roll:围绕X轴旋转。Pitch:围绕Y轴旋转。Yaw:围绕Z轴旋转。
double cy = cos(yaw * 0.5);
double sy = sin(yaw * 0.5);
double cp = cos(pitch * 0.5);
double sp = sin(pitch * 0.5);
double cr = cos(roll * 0.5);
double sr = sin(roll * 0.5);
Quaterion.x = cr * cp * cy + sr * sp * sy;

因为它不够直观,可读性较差,所以Editor Inspector显示的是Quaterion类的eulerAngles属性,它主要是让你看的。当你修改它的值时,虽然感觉像是用一个拉杆一样旋转一个物体,而实际上背后是在进行"欧拉角=>四元数"连续自动转换。

正确的手动旋转物体的操作方式是在场景中转动物体。例如,我们在场景中选择处在万向锁的角度(90,0,0)的plane,通过球体手动旋转它,可以看到欧拉角的一个非线性变化。

(图4:当plane的欧拉角x=90时,增加x,它的y和z会变为90)

这样我们把"欧拉角=>四元数"变为了"四元数=>欧拉角",场景中旋转物体,发生线性变化的是它的四元数的值,规避了欧拉角动态旋转出现万向锁的问题,然后Unity会将四元数转化为欧拉角并显示在inspector内。

这也说明万向锁其实并没有锁死任何角度,只是在这一动态问题中的某些角度,欧拉角会丧失人类可读的直观性。因此,直接正确的修改欧拉角也不是不可以解决万向锁,例如上面案例中,我们只需在物体转到(90,0,0)时手动将inspector内rotation的y和z修改为90,或在代码中写一些有关阈值的if else即可继续任意旋转。但一个平滑万能的旋转,并可通过编程简单实现,对于与动画和编程高度结合的3D软件来讲至关重要,Unity在底层必然要用四元数代替欧拉角。

由上所述四元数的非直观性质,在代码中,官网也不建议在脚本中直接分别修改四元数的xyzw,最好是对它们进行整体修改。

(图5:Quaterion的官网资料)

这里还要说一下欧拉角到四元数的转换函数。Quaterion.Eular()函数将一个欧拉角作为参数返回一个四元数,例如trans.rotation = Quaterion.Eular(new Vector3(0,30,0))。该行代码可以很直观的理解为“trans在原点围绕y轴旋转30度”(1)。使用这个函数直接修改transform有点像直接修改Editor Inspector内的Rotation,效果与直接修改Quaterion类的eulerAngles属性相同。

Quaterion.Eular()的C++实现代码应该就是如下(2):

struct Quaternion
{double w, x, y, z;
};
//Roll:围绕X轴旋转。Pitch:围绕Y轴旋转。Yaw:围绕Z轴旋转。
Quaternion ToQuaternion(double yaw, double pitch, double roll) // yaw (Z), pitch (Y), roll (X)
{// Abbreviations for the various angular functionsdouble cy = cos(yaw * 0.5);double sy = sin(yaw * 0.5);double cp = cos(pitch * 0.5);double sp = sin(pitch * 0.5);double cr = cos(roll * 0.5);double sr = sin(roll * 0.5);Quaternion q;q.w = cr * cp * cy + sr * sp * sy;q.x = sr * cp * cy - cr * sp * sy;q.y = cr * sp * cy + sr * cp * sy;q.z = cr * cp * sy - sr * sp * cy;return q;
}

如果需要动态旋转一个物体,还有Rotate(),RotateAround()等函数,我们都可以使用欧拉角作为参数。

如上,Unity使用四元数维护一个transform的角度状态,规避了物体动态旋转的万向锁问题。以四元数为基础,上层保留了欧拉角的一定角色,方便用户使用和理解。唯一容易发生迷惑的地方就是在Editor inspector面板显示的transform的rotation属性。如果用户直接对它进行拉杆式操作连续旋转一个物体就有可能会观察到一个万向锁的case,但inspector的rotation更适用于用户观察物体状态或者静态赋值,正确的手动旋转物体应该在场景scene中进行,直接修改四元数。


参考:
(1):https://docs.unity3d.com/ScriptReference/Quaternion.Euler.html
(2):https://en.wikipedia.org/wiki/Conversion_between_quaternions_and_Euler_angles

简单研究Unity中的万向锁和欧拉角以及四元数相关推荐

  1. creator中关于旋转所使用的欧拉角和四元数

    概念 四元数,欧拉角,变换矩阵,这三者是可以相互转化的,但并不是说变换矩阵等于欧拉角或者变换矩阵等于四元数,欧拉角或者说四元数他们只是旋转数据,而一个矩阵是一个空间坐标系,旋转的数据只是构造的它的一部 ...

  2. 简单理解Unity中叉乘的方向

    由于Unity使用的是左手坐标系,所以我们这里采用左手定则来表示坐标系,如图 拇指方向为X轴正向,食指为Y轴正向,中指为Z轴正向 在叉乘计算方向的时候利用好这三根手指,可以很好的判断叉乘方向. 在调用 ...

  3. C++ Eigen 库中旋转向量、旋转矩阵、欧拉角、四元数的定义及互相转换

    今天看师兄写的使用力反馈设备操作机械臂的代码,里边涉及到了Eigen 库中的旋转变换,表征旋转变换的有旋转向量Eigen::AngleAxisd.欧拉角Eigen::Vector3d.旋转矩阵Eige ...

  4. Unity 旋转欧拉角及万向锁问题

    先说明unity 3D欧拉角的旋转顺序(父子关系)是y-x-z.即旋转y轴x和z轴都变,旋转x轴只有z轴变化,旋转z轴其它轴不变. 模型坐标系 又称物体坐标系. 与特定的物体关联,每个物体都有自己特定 ...

  5. [视觉SLAM十四讲]学习笔记2-关于欧拉角和万向锁

    [视觉SLAM十四讲]学习笔记2-关于欧拉角和万向锁 1 欧拉角 1.1 维基百科定义 1.2 ZYX欧拉角与rpy角 2 万向锁 2.1 万向节 2.1 万向锁的产生 1 欧拉角 1.1 维基百科定 ...

  6. 【OpenGL_02】欧拉角、万向锁、四元数

    文章目录 欧拉角 万向锁 四元数 四元数的矩阵乘法及其可易性 蜕变矩阵 四元数的矩阵乘法 欧拉角 引用博客欧拉角与旋转矩阵的转换关系 欧拉角就是我们日常生活中常用的表示旋转的三维向量的乘积. 在Uni ...

  7. Unity3D 万向锁问题

    Unity3D 万向锁问题 1.问题 描述 在 unity3D中,对欧拉角的旋转顺序为Y-X-Z. 那么我们可以通过一个Cube来直观看到这种现象. 创建一个Cube,我们只要按照 Y-X-Z顺序 操 ...

  8. 旋转矩阵|万向锁|四元数

    文章目录 旋转公式 2D 3D 万向锁 四元数 3D旋转公式 Reference 旋转公式 2D 矩阵形式 v′=[cos(θ)−sin(θ)sin(θ)cos(θ)]vv' = \begin{bma ...

  9. unity 旋转欧拉角 万向锁 解释

    万向锁 一直困惑我很久....原因出在这里,我以为欧拉角旋转是以模型坐标(齐次坐标系)为旋转轴.问题就来了,无论旋转那个轴,其它两个轴也会相应的变化,下面看图: 根据上面的说明两个旋转面(圆圈)怎么会 ...

最新文章

  1. swift string转int_Swift进阶二:基本数据类型相关
  2. Windows server 2008 处理多用户在登陆时显示问题
  3. docker挂载本地目录的方法总结:
  4. Eclipse实现hibernate反向工程:从数据库逆向生成实体类和hbm文件
  5. 8月19学习练习[两三个TableView并排显示]
  6. 深入浅出mfc_深入浅出HBase系列(一)
  7. 大年30还多少天_大美鹅老李告诉你30天的鹅需要多少温度?
  8. 怎么能把你的公司快速做大呢
  9. 团队编程项目作业4-开发文档
  10. 拜托,别再让我优化大事务了,我的头都要裂开了
  11. 一. Vue项目引入字体(思源黑体)
  12. PHP容器--Pimple运行流程浅析
  13. 2021年,各类显卡的计算能力对比,天梯图
  14. iPhone4/4s 5.1.1版本越狱后无法连接iTunes,出现0xE8000012错误的解决方法
  15. 微信公众号留言功能怎么开通?
  16. 罗技 连点 脚本_罗技发布无线版的Pro X游戏耳机,拥有20小时续航时间
  17. 清理Git提交记录最简单的方法
  18. C++ Qt高仿QQ影音视频播放器 (一)
  19. 数字电路与逻辑设计笔记1
  20. Cypress 本身启动过程的调试

热门文章

  1. python将图像变成没有颜色_python将图片设置背景颜色修改为透明色
  2. docker配置Java环境
  3. matlab画波动图像,【基于Matlab的波动方程的可视化实现最终版材料】
  4. mysql 使用if函数实现多条件动态查询
  5. zblog采集-织梦全自动采集插件-织梦免费采集插件
  6. 跨越逐梦路上的荆棘(程序猿生存指南)
  7. 《大话数据结构》读书笔记(二)
  8. ffmpeg中的时间 DTS、PTS、AV_TIME_BASE、AV_TIME_BASE_Q 介绍
  9. SAP MM 因物料有负库存导致MMPV开账期失败问题之对策
  10. sql中的coalesce