博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
四元数运算例子
阅读量:6693 次
发布时间:2019-06-25

本文共 3050 字,大约阅读时间需要 10 分钟。

hot3.png

这里不介绍四元数的历史了,我只说明下他是一个超复数(请不要被这个名字吓倒了)这儿你不需要了解啥是超复数,只要你了解啥是向量,那么你就能读懂这篇文章了。

四元数的一种表示方式是: Q = xi + yj + zk + w
这儿i,  j,  k就可以看成3D 空间的3个坐标柱向量。  基于四元数的这种表示方式所以很直接他就可以被表示成一个标量w
再加上一个3维向量了。
Q = [w, v]
这里 v = xi + yj + zk
自然有一定编程经验的人会立即把这个小家伙用结构体来表示了如下:
struct quaternion
{
       double x, y, z, w;
}
这里我们没必要知道四元数的加减运算法则了,现在为了解决我们的目标(创建基于四元数的摄像机)我们第一步需要知道怎样把他标准化
。四元数的标准化和向量的标准化是一样的。目的都是将模变成1. |Q| = sqrt(w^2 + x^2 +y^2 + z^2)
下面是代码
double getLength(quaternion quat)
{
       return sqrt(quat.x * quat.x + quat.y * quat.y + quat.z * quat.z + quat.w * quat.w);
}
(呵呵,长度计算这样给大家写出来请大家不要告我小瞧大家的智商)
那么得到标准化得四元数自然是水到渠成的事了 Q* = Q/|Q| = [w/|Q|, v/|Q|];
这儿是代码:
quaternion normalize(quaternion quat)
{
       double Length = length(quat);
       quat.x /= Length;
       quat.y /= Length;
       quat.z /= Lenght;
       quat.w /= Length;
       return quat;
}
下来我们需要知道怎样计算一个四元数的共轭四元数, 共轭四元数暂且用Q' 来代替吧。
Q' = [w, -v]
这儿是计算共轭四元数的代码:
quaternion conjugate(quaternion quat)
{
       quat.x = -quat.x;
       quat.y = -quat.y;
       quat.z = -quat.z;
       quat.w = -quat.w;
       return quat;
}
下面的事情就是看看怎样计算四元数的成绩了,这个貌似有点小小的复杂。
假设有四元数C , A , B 现在我们要计算 C= A* B;
具体的运算规则如下:
C.x = |A.w * B.x + A.x * B.w + A.y * B.z - A.z * B.y|
C.y = |A.w * B.y - A.x * B.z + A.y * B.w + A.z * B.x|
C.z = |A.w * B.z + A.x * B.y - A.y * B.x + A.z * B.w|
C.w = |A.w * B.w - A.x * B.x - A.y * B.y - A.z * B.z|
这儿是代码:
quaternion mult(quaternion quat)
{
      quaternion C;
      C.x = A.w*B.x + A.x*B.w + A.y*B.z - A.z*B.y;
      C.y = A.w*B.y - A.x*B.z + A.y*B.w + A.z*B.x;
      C.z = A.w*B.z + A.x*B.y - A.y*B.x + A.z*B.w;
      C.w = A.w*B.w - A.x*B.x - A.y*B.y - A.z*B.z;
      return C;
}
主要的运算讲完了,下面就要进入正题了。 现在我们是要创建摄像机了请注意。很明显在3D空间里面代表一个摄像机
就需要对这个摄像机进行定位,定向了。所以为了精确的创建摄像机,我们使用3个3维向量来代表摄像机的位置Position,观察方向View,
和摄像机的UP(这个没必要介绍吧)。对于一个第一人称的摄像机来说,我们现在就只需要考虑摄像机的旋转问题了,
使用四元数我们就可以把一个向量绕任意的柱旋转, 为了达到这个目的我们就需要首先将View向量转化成四元数,然后定义一个旋转四元数,最后应用这个旋转四元数到View四元数上就行了。下面请看具体的步骤。
为了获得View四元数,我们就要使用[w, v]来代表了。当然标量w 就设成0了。v 很显然就是View向量了。
因此四元数V(我们转化的View四元数) V = [0, View]
然后正如上面所说我们下面就要创建那个旋转四元数了,为了创建这个四元数,你就得明确你要绕哪个向量旋转了,还有旋转的角度。我们把这个柱向量(就是绕哪个向量旋转)叫做A,把旋转的角度用theta表示,那么任务即将完成了,我们创建的旋转四元数R就可以表示成:
3DVector A = [x, y , z];
R.x = A.x * sin(theta/2);
R.y = A.y * sin(theta/2);
R.z = A.z * sin(theta/2);
R.w = cos(theta/2);
下来就要进行旋转计算了:
先看看我们现在知道的量:
1. 我们有3维的View向量, 我们有View四元数 V = [0, View]
2. 假设我们要绕A向量来旋转theta度,我们有旋转四元数R来定义这次旋转。
3. 记住旋转完后我们得到的东西仍然是一个View四元数,这个自然是个新的View四元数了。我们假设他是W
所以旋转操作就是:
W = R * V * R'
这里R是旋转四元数, V是View四元数, R' 是旋转四元数R的共轭四元数(其运算前面以讲了)
现在我们仅仅提取新的View四元数W的向量部分。 NewView = [W.x, W.y, W.z]
所以综上所述我们就可以创建函数来进行旋转变换:
void RotateCamera(float angle, float x, float y, float z)
{
       quaternion temp, quat_view, result;// quat_view是View四元数, temp是旋转四元数
     //  下面的三行使用旋转柱和角度计算旋转四元数
       temp.x = x * sin(angle/2);
       temp.y = y * sin(angle/2);
       temp.z = z * sin(angle/2);
       temp.w = cos(angle/2);
       // 计算View四元数
       quat_view.x = View.x; // View 是View向量
       quat_view.y = View.y;
       quat_view.z = View.z;
       quat_view.w = 0;
       //  进行旋转变换
       result = mult(mult(temp, qaut_view), conjugate(temp));
       // 得到新的View向量
       View.x = result.x;
       View.y = result.y;
       View.z = result.z;
}
至此完成。相信有一定OpenGL或D3D基础的人,很容易就把这个翻译成代码了。

转载于:https://my.oschina.net/u/3773235/blog/1614905

你可能感兴趣的文章
《Servlet、JSP和Spring MVC初学指南》——第2章 会话管理 2.1URL重写
查看>>
Docker 之 构建镜像
查看>>
《微信小程序开发入门精要》——第2章,第2.5节水平排列对齐方式
查看>>
《异构信息网络挖掘: 原理和方法》—— 第2章 基于排名的聚类 2.1 概述
查看>>
[案例]网易云音乐的个性化推荐
查看>>
《C++编程惯用法——高级程序员常用方法和技巧》——1.1 有关电话号码的抽象模型...
查看>>
《Java线程与并发编程实践》—— 2.5 练习
查看>>
《HTML5+JavaScript动画基础》——第1章 动画的基本概念 1.1动画
查看>>
《实践者的研究方法》—— 第3章 软件工程3.2 定义框架活动
查看>>
《QTP自动化测试权威指南(第二版)》—第2章2.3节搜索标签(Search Tab)
查看>>
《iOS 8应用开发入门经典(第6版)》——第1章,第1.3节运行iOS应用
查看>>
Activiti实战. 2.3配置文件介绍
查看>>
《ELK Stack权威指南(第2版)》一1.5 长期运行方
查看>>
《树莓派开发实战(第2版)》——2.4 为树莓派配置网络名称
查看>>
SEO优化网站排名是靠什么支撑的?
查看>>
spark源码分析之Checkpoint的过程
查看>>
openstack 与 ceph (架构)
查看>>
满满的技术干货!Spark顶级会议Apache Spark Summit精华讲义分享
查看>>
Android热修复技术——QQ空间补丁方案解析(2)
查看>>
云开雾散,带你看看大数据行业
查看>>