首页 >后端开发 >C++ >如何有效计算 2D 和 3D 空间中两个向量之间的顺时针角度?

如何有效计算 2D 和 3D 空间中两个向量之间的顺时针角度?

Patricia Arquette
Patricia Arquette原创
2024-11-21 07:44:09874浏览

How can I efficiently calculate the clockwise angle between two vectors in both 2D and 3D space?

高效计算向量之间的顺时针角度

传统上,需要利用点积计算两个向量之间的角度,点积决定了向量之间的内角范围为 0 至 180 度。然而,这种方法在确定角度与其补角之间的适当结果时存在挑战。

是否有更直接的方法来计算顺时针角度?

2D 案例

与点积与角度余弦的关系类似,行列式与其正弦成正比。通过结合这种关系,我们可以计算角度如下:

dot = x1 * x2 + y1 * y2  # Dot product between [x1, y1] and [x2, y2]
det = x1 * y2 - y1 * x2  # Determinant
angle = atan2(det, dot)  # atan2(y, x) or atan2(sin, cos)

计算出的角度的方向与坐标系的方向一致。在左手坐标系中,x 指向右侧,y 指向下方,顺时针角度将产生正值。相反,在 y 指向上方的数学坐标系中,结果反映逆时针角度,这是数学中的惯例。交换输入向量的顺序会改变符号,从而可以灵活地修改结果的符号。

3D Case

在三维空间中,任意向量定义自己的轴旋转垂直于两者。由于该轴没有固定方向,因此无法唯一确定旋转角度的方向。常见的约定是指定正角度并对齐轴以适应此约定。在这种情况下,归一化向量的点积足以进行角度计算:

dot = x1 * x2 + y1 * y2 + z1 * z2  # Between [x1, y1, z1] and [x2, y2, z2]
lenSq1 = x1 * x1 + y1 * y1 + z1 * z1
lenSq2 = x2 * x2 + y2 * y2 + z2 * z2
angle = acos(dot / sqrt(lenSq1 * lenSq2))

嵌入 3D 的平面

对于约束在具有已知法线的平面内的向量向量 n,需要考虑特定情况。旋转轴与n 重合,n 的方向固定轴的方向。在这种情况下,我们可以修改上面的 2D 计算,将 n 包含在行列式中,将其转换为 3x3 矩阵:

dot = x1 * x2 + y1 * y2 + z1 * z2
det = x1 * y2 * zn + x2 * yn * z1 + xn * y1 * z2 - z1 * y2 * xn - z2 * yn * x1 - zn * y1 * x2
angle = atan2(det, dot)

为了使此计算有效,法向量 n 必须标准化为单位长度。

或者,行列式可以表示为三元组产品:

det = n · (v1 × v2)

这种方法在某些 API 中可能更容易实现,并提供对底层机制的深入了解:叉积与角度的正弦值成正比,并且垂直于平面,这意味着它是一个倍数的 n.因此,点积本质上是测量应用了正确符号的向量的长度。

范围 0 – 360°

大多数 atan2 实现返回该范围内的角度[-π, π](以弧度为单位)或 [-180°, 180°](以度为单位)。要获得 [0, 2π] 或 [0°, 360°] 范围内的正角度,可以应用以下变换:

dot = x1 * x2 + y1 * y2  # Dot product between [x1, y1] and [x2, y2]
det = x1 * y2 - y1 * x2  # Determinant
angle = atan2(det, dot)  # atan2(y, x) or atan2(sin, cos)

或者,以下表达式避免区分大小写:

dot = x1 * x2 + y1 * y2 + z1 * z2  # Between [x1, y1, z1] and [x2, y2, z2]
lenSq1 = x1 * x1 + y1 * y1 + z1 * z1
lenSq2 = x2 * x2 + y2 * y2 + z2 * z2
angle = acos(dot / sqrt(lenSq1 * lenSq2))

此校正技术不限于此特定问题,而是可以应用于涉及 atan2 的任何场景。

以上是如何有效计算 2D 和 3D 空间中两个向量之间的顺时针角度?的详细内容。更多信息请关注PHP中文网其他相关文章!

声明:
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn