学习四元数

欧拉角会遇到的问题

  • 旋转顺序很敏感:先绕 Y 再绕 X,和先绕 X 再绕 Y,结果完全不同
  • "轴会变味/不好跟手":当物体已经转过一圈后,你再做"上下",实际绕的轴可能不是你以为的那个
  • 极端情况下会"卡住"(万向节死锁,gimbal lock):两个旋转轴重合,少了一个自由度

四元数

为了解决使用欧拉角中会遇到的问题,我们使用四元数:

四元数的核心价值:把旋转当成一个整体来累计,不会出现万向节死锁,并且叠加旋转非常自然。

欧拉角是三个旋转轴,四元数可以当作是一个旋转轴,这个旋转轴每次都是在变化的。

四元数可以写作:
q = ( x , y , z , w ) q=(x, y, z, w) q=(x,y,z,w)

理解:

  • 一个单位四元数(长度=1)可以唯一表示一个 3D 旋转
  • 它等价于"轴-角"表示法:绕单位轴 n = (nx, ny, nz) 旋转角度 θ

与欧拉角的关系:

  • w = cos ⁡ ( θ / 2 ) w=\cos(\theta / 2) w=cos(θ/2)
  • ( x , y , z ) = n ∗ sin ⁡ ( θ / 2 ) (x, y, z)=n * \sin(\theta / 2) (x,y,z)=n∗sin(θ/2)

使用四元数

  1. 创建一个"绕轴旋转"的四元数
cpp 复制代码
QQuaternion q = QQuaternion::fromAxisAndAngle(axis, angleDeg);
  1. 叠加旋转
    如果当前姿态 cur,鼠标产生一个增量旋转 dq,更新是:
cpp 复制代码
cur = (dq * cur).normalized();
  1. 用四元数旋转向量(得到"当前物体的局部轴")
cpp 复制代码
QVector3D right = cur.rotatedVector(QVector3D(1,0,0));
相关推荐
故事和你911 天前
洛谷-数据结构1-1-线性表1
开发语言·数据结构·c++·算法·leetcode·动态规划·图论
脱氧核糖核酸__1 天前
LeetCode热题100——53.最大子数组和(题解+答案+要点)
数据结构·c++·算法·leetcode
脱氧核糖核酸__1 天前
LeetCode 热题100——42.接雨水(题目+题解+答案)
数据结构·c++·算法·leetcode
王老师青少年编程1 天前
csp信奥赛C++高频考点专项训练之贪心算法 --【线性扫描贪心】:数列分段 Section I
c++·算法·编程·贪心·csp·信奥赛·线性扫描贪心
王老师青少年编程1 天前
csp信奥赛C++高频考点专项训练之贪心算法 --【线性扫描贪心】:分糖果
c++·算法·贪心算法·csp·信奥赛·线性扫描贪心·分糖果
leaves falling1 天前
C++模板进阶
开发语言·c++
无敌昊哥战神1 天前
【保姆级题解】力扣17. 电话号码的字母组合 (回溯算法经典入门) | Python/C/C++多语言详解
c语言·c++·python·算法·leetcode
脱氧核糖核酸__1 天前
LeetCode热题100——238.除了自身以外数组的乘积(题目+题解+答案)
数据结构·c++·算法·leetcode
ouliten1 天前
C++笔记:std::invoke
c++·笔记
j_xxx404_1 天前
C++算法:哈希表(简介|两数之和|判断是否互为字符重排)
数据结构·c++·算法·leetcode·蓝桥杯·力扣·散列表