【Unity编程】欧拉角与万向节死锁(图文版)

万向节死锁(Gimbal Lock)问题

上文中曾经说过,欧拉旋转的顺规和轴向定义,自然造就了“万向节死锁”问题。本文主要来探索它自然形成的原因。

陀螺仪

首先,我们来了解Gimbal 究竟是个什么玩意儿。下面来自维基百科中关于Gimbal的一段引述:



平衡环架(英语:Gimbal)为一具有枢纽的装置,使得一物体能以单一轴旋转。由彼此垂直的枢纽轴所组成的一组三只平衡环架,则可使架在最内的环架的物体维持旋转轴不变,而应用在船上的陀螺仪、罗盘、饮料杯架等用途上,而不受船体因波浪上下震动、船身转向的影响。



上图就是一个Gimbal装置了,它是一个陀螺仪。中间有一根竖轴,穿过一个金属圆盘。金属圆盘称为转子,竖轴称为旋转轴。转子用金属制成,应该是了增加质量,从而增大惯性。竖轴外侧是三层嵌套的圆环,它们互相交叉,带来了三个方向自由度的旋转。

看着不停转来转去,有点晕,接下来看两个静态的。这两张图来自百度百科。

中文注释:

其中Gimbal只代表陀螺仪装置中的平衡环,显然维基百科上将它解释成“平衡环架”更为合理。

Pitch、Yaw、Roll

在解释陀螺仪的工作原理之前,我先介绍一些转动的术语。在飞行器的航行中,进行XYZ三个方向旋转的旋转有专业的术语,见下图:

沿着机身右方轴(Unity中的+X)进行旋转,称为pitch,中文叫俯仰

沿着机头上方轴(Unity中的+Y)进行旋转,称为Yaw,中文叫偏航

沿着机头前方轴(Unity中的+Z)进行旋转,称为Roll,中文叫桶滚

陀螺仪的工作原理

我们知道陀螺仪使用来测量平衡和转速的工具,在载体高速转动的时候,陀螺仪始终要通过自我调节,使得转子保持原有的平衡,这一点是如何做到的?带着这个问题,我们来看一下这个古老而又神秘的装置的工作原理。

为了解释清楚问题,我自己画了一个简单的陀螺仪示意图。(金属圆盘我就省略了,丑点儿也就别管了。。)

这里,我把三个Gimbal环用不同的颜色做了标记,底部三个轴向,RGB分别对应XYZ。

假设现在这个陀螺仪被放在一艘船上,船头的方向沿着+Z轴,也就是右前方。

  • 现在假设,船体发生了摇晃,是沿着前方进行旋转的摇晃,也就是桶滚。由于转子和旋转轴具有较大的惯性,只要没有直接施加扭矩,就会保持原有的姿态。由于上图中绿色的活动的连接头处是可以灵活转动的,此时将发生相对旋转,从而出现以下的情形:

  • 再次假设,船体发生了pitch摇晃,也就是俯仰。同样,由于存在相应方向的可以相对旋转的连接头(红色连接头),转子和旋转轴将仍然保持平衡,如下图:

  • 最后假设,船体发生了yaw摇晃,也就是偏航,此时船体在发生水平旋转。相对旋转发生在蓝色连接头。如下图:

最终,在船体发生Pitch、Yaw、Roll的情况下,陀螺仪都可以通过自身的调节,而让转子和旋转轴保持平衡。

陀螺仪中的万向节死锁

现在看起来,这个陀螺仪一切正常,在船体发生任意方向摇晃都可以通过自身调节来应对。然而,真的是这样吗?

假如,船体发生了剧烈的变化,此时船首仰起了90度(这是要翻船的节奏。。。。),此时的陀螺仪调节状态如下图:

此时,船体再次发生转动,沿着当前世界坐标的+Z轴(蓝色轴,应该正指向船底)进行转动,那么来看看发生了什么情况。

现在,转子不平衡了,陀螺仪的三板斧不起作用了。它失去了自身的调节能力。那么这是为什么呢?

之前陀螺仪之所以能通过自身调节,保持平衡,是因为存在可以相对旋转的连接头。在这种情况下,已经不存在可以相对旋转的连接头了。

那么连接头呢?去了哪里?显然,它还是在那里,只不过是,连接头可以旋转的相对方向不是现在需要的按着+Z轴方向。从上图中,我们清楚地看到:

  • 红色连接头:可以给予一个相对俯仰的自由度。
  • 绿色连接头:可以给予一个相对偏航的自由度。
  • 蓝色连接头:可以给予一个相对偏航的自由度。

没错,三个连接头,提供的自由度只对应了俯仰和偏航两个自由度,桶滚自由度丢失了。这就是陀螺仪上的“万向节死锁”问题。

用小程序来重现万向节死锁问题

首先,预设一下接下来的欧拉角变化顺序。见下图:

上图中,红色框内的部分的列表,记录了接下来欧拉角的增长变化过程。即它会从(0,0,0)变化到(90,0,0),再变化到(90,90,0),再变化到(90,180,0),再变化到(90,180,90),再变化到(90,180,180)。下图是变化的过程演示。

现在可以看到:

- 当先执行X轴旋转90度,此时在执行Pitch(俯仰)变化。

- 再在Y轴进行变化0-180度,此时在执行相对自身的Roll(桶滚)变化。

- 再在Z轴进行变化0-180度,此时仍在执行相对自身的Roll(桶滚)变化。

这里所说的俯仰、桶滚、偏航都是相对自己局部坐标系的。这与上述的陀螺仪中出现的问题是一样的,万向节死锁。也就是尽管欧拉角在XYZ三个轴向进行进动(持续增长或者减少),但是影响最终的结果,只对应了两个轴向。

死锁的过程解析

《Unity中欧拉旋转》一文中我曾提到,是欧拉角顺规和轴向的定义方式,造就了“万向节死锁”问题的自然形成。通过上述的例子,这里作个详细解释。

首先我们知道,由于Unity中欧拉旋转的顺规的定义,围绕Z轴的进动最先执行,所以,Z轴是“严格保护”的一个轴,就是说,当先沿着Z轴进行进动时,无论此时的XY是什么值,最终的结果,围绕Z轴的进动始终造成相对自身执行桶滚变化。

然而X、Y轴就不同了,我们先不考虑Y轴,假设其一直为0,先说X轴。如果Z轴也是保持为0,那么围绕X轴进动,最终的影响是预期的俯仰变化。如下图:

然而当Z轴为90度时,围绕X轴进动变成了偏航变化,如下图:

也就是说,欧拉角的X轴进动造成最后的变化结果,受到到了预先执行的Z轴进动的影响,它仍然会造成某个相对自身的轴向的变化,但是结果不唯一;同样,欧拉角的Y轴进动,则受到了Z轴和X轴的影响,结果更加不唯一。

然而,以上的过程执行,都是严格遵守欧拉角的顺规和轴向定义的。某些时刻,这种不确定的结果,就可能造成某个轴向自由度的丢失。

就拿下图来说:

欧拉角Z轴的进动,最先执行,造成桶滚,这个没问题。

欧拉角Y轴的进动,最后执行,造成沿着欧拉旋转前的Y轴旋转,这也是根据定义执行。然而现在这种沿着Y轴的旋转,同样也被映射到了物体的桶滚变化。

总结

总结来说,欧拉角的“万向节死锁”问题,是由于欧拉旋转定义本身造成的。这种围绕选旋转前固定轴的先Z、再X、再Y的旋转操作,与其最终所预期的三个轴向可以旋转的结果并非一定是一对一的映射。某些情况下是多对一的映射,造成一些旋转自由度的缺失,也就是“死锁”。

建议

对于写代码来说,你直接去改变Transform的欧拉角显然是不合适的,通过本文也可以看到,这种结果几乎是不可预测的。但是某些情况下,却是可以预期的,就是你仅仅在一个轴向进动,其它两个轴向保持为0,此时有效,并且直接修改欧拉角的代码效率应该是比较高的。

时间: 2024-10-28 21:41:06

【Unity编程】欧拉角与万向节死锁(图文版)的相关文章

欧拉角之万向节死锁

最近阅读<3D数学基础:图形与游戏开发>,看到欧拉角一节,对万向锁感到难以理解.先看一段视频. 欧拉旋转 什么是欧拉角?欧拉角是一种描述物体旋转的方式,使用三个值表示物体在三个互相垂直的轴上的旋转.这里的三个轴是任意相互垂直的三个轴,旋转顺序也是随意的.我们可以按照一定的约定来对物体进行欧拉角旋转.此处使用heading(沿Y轴旋转)—pitch(X轴)—bank(Z轴)的旋转顺序,旋转正方向为从坐标轴正方向望去的顺时针方向. 什么是别名?用欧拉角来表示方位(orientation)的缺点是指

【Unity编程】四元数(Quaternion)与欧拉角

欧拉旋转.四元数.矩阵旋转之间的差异 除了欧拉旋转以外,还有两种表示旋转的方式:矩阵旋转和四元数旋转.接下来我们比较它们的优缺点. 欧拉角 优点:三个角度组成,直观,容易理解. 优点:可以进行从一个方向到另一个方向旋转大于180度的角度. 弱点:死锁问题. 前面<[Unity编程]欧拉角与万向节死锁(图文版)>已经介绍过万向节死锁问题. 四元数 内部由四个数字(在Unity中称为x,y,z和w)组成,然而这些数字不表示角度或轴,并且通常不需要直接访问它们.除非你特别有兴趣深入了解四元数学,你只

欧拉角万向节死锁与四元数的相关资料

欧拉角和四元数都可以用来描述物体的姿态.欧拉角的万向节死锁理解稍微奇怪.四元数原理的理解较为复杂.欧拉角和四元数之间可以做转换. 这里收集了若干资料已做备忘. 欧拉角: 1. 欧拉角基本概念 2.欧拉角的万向节死锁(万向节死锁的理解,不错) 3.万向节死锁视频讲解(万向节死锁的视频讲解,老外) 4. 关于万向节死锁(Gimbal Lock)(万向节死锁的一个解释,还好) 5. 万向节死锁实验讲解视频(老外很用心去解释死锁的概念) 四元数资料: 1. 彻底搞懂四元数(这里有四元数的理解,旋转矩阵,

blender 万向坐标轴, 欧拉角 , 3d软件的实际坐标系统 , 陀螺仪做为坐标轴 ,万向节死锁的解释

https://www.cnblogs.com/delphi-xe5/p/12340038.html https://www.cnblogs.com/delphi-xe5/p/12340039.html https://www.cnblogs.com/delphi-xe5/p/12340053.html https://www.cnblogs.com/psklf/p/5656938.html 参考上面 四个资料 .总结自己的理解 1.  欧拉角  分 静态  和动态.  静态就是 旋转的时候坐标

【Unity编程】Unity中关于四元数的API详解

Unity中关于四元数的API详解 Quaternion类 Quaternion(四元数)用于计算Unity旋转.它们计算紧凑高效,不受万向节锁的困扰,并且可以很方便快速地进行球面插值. Unity内部使用四元数来表示所有的旋转. Quaternion是基于复数,并不容易直观地理解. 不过你几乎不需要访问或修改单个四元数参数(x,y,z,w); 大多数情况下,你只需要获取和使用现有的旋转(例如来自"Transform"),或者用四元数来构造新的旋转(例如,在两次旋转之间平滑插入). 大

万向节死锁(Gimbal Lock)

网上也有不少关于万向节死锁的文章,但写的都比较繁琐,这个问题只需知道个概念即可. 在机器人学中这个问题一般被称为动力学奇点(Kinematic singularity).如图1,可以看到连接坐标轴的有三个旋转关节,旋转关节1是绕X6轴旋转的,旋转关节2是绕Y6轴旋转的,旋转关节3是绕Z6轴旋转的.我们现在可以将三个旋转关节看成一个整体,相当于手腕,因为手腕也就是三个自由度.那么为什么把这个称为动力学奇点呢?我们做个小小的变化,我们将整个结构绕Y6轴向外旋转90度.这个时候旋转关节1和旋转关节2是

PyCharm5.0.2最新版破译注册激活码(图文版)

PyCharm5.0.2最新版破解注册激活码(图文版) 下载PyCharm http://download-cf.jetbrains.com/python/pycharm-professional-5.0.2.exe 安装PyCharm 设置激活服务器  http://idea.lanyus.com

关于使用WORD+OUTLOOK 群发的教程(图文版)

关于使用WORD+OUTLOOK 群发的教程(图文版),需要的工具:一般Microsoft Office 2007 以上版本含有此教程需要的两个工具OUTLOOK 和WORD.(下图) 下面是利用word 群发邮件的方法步骤. 在此之前要求业务员本身有良好的整理客户资料的习惯.Excel 表格要登记客户的详细信息.如图: 在你有完善的客户列表之后,就相当于你有了一份完善的数据库.下面打开新建的word文档,来进行下一步. 打开word 文档之后,写上你要发出去的开发信.以下作为举例. 完成开发信

VMware Workstation安装CentOS7.0 图文版

1 前期准备: 下载虚拟机和CentOS安装源 VMware-workstation-full-10.0.3-1895310 CentOS-7.0-1406-x86_64-DVD.iso 以下是过程截图及少许说明. 2 虚拟机配置 -1 打开虚拟机 -2 -3 -4 -5 -6 -7 也可以使用默认的处理器数量:1 -8 我修改了此虚拟机内存为2G -9 -10 -11 -12 -13 根据需要分配大小 -14 -15 虚拟机设置完成 -16 终览 3 安装CentOS7.0 -1 -2 -3