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

article/2025/7/21 3:25:12

欧拉角是欧拉在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


http://chatgpt.dhexx.cn/article/1TaVXeEp.shtml

相关文章

Unity万向锁

结论: 当第二轴旋转90时会产生万向锁,万向锁只发生在程序中,现实世界不会发生。 发生万向锁的底层原因是由于程序中会指定旋转顺序,且前后顺序为父子包含关系。 旋转顺序:因为同样一组数据,例如{X:10&…

欧拉角与万向锁—深度解读万向锁

文章目录 问题描述:Z轴不生效(轴向重合)轴向为什么会重合初态二次变换总结:解决办法: 相信很多人听说过欧拉角与万向锁,主要问题是欧拉角为什么会产生万向锁? 有矩阵解读法,但是不够…

关于万向锁(Gimbal lock)问题的理解及解决

有时间更新一下理解,现在回过头来看写的东西真是垃圾哈哈哈哈哈~~~~ 1. Gimbal lock的理解 万向锁(Gimbal lock): 一旦选择90作为pitch角,就会导致第一次旋转和第三次旋转等价,整个旋转表示系统被限制在只…

详解万向锁

最近在看高翔老师的slam十四讲时,对万向锁这个概念不是很清晰,于是从网上查了不少博客文档,最终明白了万向锁这个概念。 在此记录一下。 解释万向锁之前首先明白一下什么是欧拉角。 什么是欧拉角? 用一句话说,欧拉角…

java uuid多少位_UUID用在MySQL中,性能到底好不好?

如果你在网上快速的做一个关于 UUID 和 MySQL 的搜索,你会得到相当多的结果。以下是一些例子: 存储 UUID 和 生成列在 MySQL 中存储 UUID 的值说明 InnoDB 中的主键模型及其对磁盘使用的影响主键选型之战 UUID vs. INTGUID / UUID 的性能突破到底需不需要…

UUID 全世界的唯一id!(唯一的接口标识符)

UUID(Universally Unique IDentifier)是一个128位数字的唯一标识。RFC 4122描述了具体的规范实现。本文尝试从它的结构一步步分析为什么它能做到唯一性?及各个版本的使用场景。 Format UUID使用16进制表示,共有36个字符(32个字母数字4个连接符"-&…

瓦片地图加载

OpenLayers之加载多源数据四:万能瓦片地图加载秘籍 不睡觉的怪叔叔 2018-08-02 13:46:59 7175 收藏 17 分类专栏: 开源GIS 目录 加载简单的瓦片地图 瓦片地图之高德地图 瓦片地图之百度地图 重新定义OpenLayers3的瓦片坐标系 分析瓦片地图坐标系…

QGIS 上加载瓦片地图(高德、ESRI、OSM等)

1. 在QGIS左侧的浏览面板中,选择XYZ Tiles节点,右键新建连接。在弹出的窗口中填入名称和瓦片地图的网址http://webrd02.is.autonavi.com/appmaptile?langzh_cn&size1&scale1&style7&x{x}&y{y}&z{z} ,保存。 2.将浏览…

QGIS二次开发:加载XYZ Tiles形式的瓦片地图

1.前言 QGIS官方提供的QGIS.exe可以加载远程地图提供商提供的基于URL格式的XYZ Tiles形式的瓦片地图。如:https://tile.openstreetmap.org/{z}/{x}/{y}.png ,开发者如果不能联网,也可以创建离线地图服务。如下:打开QGIS 3.26.2.exe,选择xyz Tiles节点,单击右键,选择“N…

Vue Mapbox 发布高德瓦片地图

一、需求来源 公司网站开发需求里需要做一个大屏看板,包含地图的功能,而且以后是要在内网使用的,也就是说不能使用在线网络。我在研究了(踩了一万个坑!!!)多个地图组件以后&#xff…

瓦片地图的入库

数据: 瓦片地图 数据库: MySQL 语言 : Java 方法: 用Java语言把瓦片地图以二进制的方式导入数据库 瓦片地图的存储格式 当我们用ArcMap切完瓦片之后, 它的存储方式是以级数,行列号的方式存储在文件夹中 级数: 行号: 列号: 一般的存储结构在数据库中存储它的 级数,行号,列…

各种瓦片地图

概述 研究了一阵子的地图,太过深奥了。写了一个小程序,可以下载瓦片地图。下面是各种瓦片图例。 瓦片地图 微软瓦片地图 图1 图2 谷歌瓦片地图 图3 图4 osm瓦片地图 图5 图6 天地图瓦片地图 图7 图8 水瓦片地图 图9 图10 后续 还…

geoserver发布TIF格式瓦片地图

一,准备tif地图 可以通过全能电子地图下载器获取地图(软件可以去淘宝下载) 二,geoserver发布tif 在geoserver安装文件夹中双击“startup.bat”打开服务(注意:为了防止端口占用,我这里更改了默…

Unity 瓦片地图

一些常见的使用类似方法绘制地图的游戏: 泰拉瑞亚: 如果有老哥玩过泰拉瑞亚的地图编辑器,其实unity瓦片地图和泰拉瑞亚地图编辑器手感类似 蔚蓝: 大名鼎鼎的蔚蓝也是如此 Untiy中制作2D俯视角游戏,往往需要使用瓦片地图…

瓦片地图下载工具

收费工具,学生党勿扰,白嫖党勿扰 收费金额:200元 MyChat:wangjianjun1018 瓦片地图下载工具 概述操作查看地图 下载链接说明后续 概述 最近花费一点时间,研究瓦片地图。在网上找了好久,有好多下载的提供商…

百度个性化瓦片地图下载

百度个性化瓦片地图下载 因为近期项目需求,客户项目在内网使用,不连接外网,因此地图不能使用在线地图,必须使用离线地图。 离线地图使用在之前的博客写过,就不在累述了。https://blog.csdn.net/weixin_42776111/arti…

Tilemap瓦片地图

可以用Tilemap工具来构建游戏世界的地图,创建瓦片地图时,Grid组件自动作为瓦片地图的父级,相比于传统使用照片搭建地图的方式,使用瓦片地图用来搭建地图可以更加迅速,而且使用传统方式搭建地图时,需要添加大量的碰撞体…

leaflet加载离线瓦片地图

首先我们要明白瓦片地图的请求原理。 其实瓦片地图并不是什么特殊的文件,就是最普通的png图片。之所以为地图,就是带有了该图片按规则组织,绘制时,按规则拼图组装而已; 打开network,查看其请求的的url就能…

瓦片地图原理

GIS介绍 地理信息系统(Geographic Information System或 Geo-Information system,GIS)有时又称为“地学信息系统”。它是一种特定的十分重要的空间信息系统。它是在计算机硬、软件系统支持下,对整个或部分地球表层&…

Unity(9)-TileMap瓦片地图

文章目录 前言相关介绍其他介绍上一篇笔记下一篇笔记 一、图片预处理二、创建调色板(TilePalette)三、创建瓦片(Tile)[1]. 方式一[2]. 方式二 四、修改瓦片(Tile)[1]. 参数介绍[2]. 修改Sprite[3]. 修改Color[4]. 修改ColliderType 五、创建瓦…