【Triangulation】三角测量

1. 三角测量定义

在三维重建中,三角测量(Triangulation)是一种基本的算法,用于利用两个摄像头从两个不同视角拍摄得到的图片中的匹配点来还原一个三维点的位置。具体方法是通过解线性方程组,找到最佳的三维点位置,使得该点在两个摄像机视角下的投影和实际观察到的二维点尽可能匹配。

2. 公式推导

这个 TriangulatePoint 函数接受两个矩阵(cam1_from_worldcam2_from_world,它们代表两个摄像机相对于世界坐标系的变换矩阵),以及两个二维向量(point1point2,它们表示在每个摄像机视角下观察到的点的坐标)。函数的目标是计算出全局坐标系中的三维点,该点在数学上应该映射到这两个二维点上。

为了进行三角化,我们利用摄像机投影矩阵的性质。一个点 P P P 在世界坐标系中的坐标可以通过摄像机的投影矩阵 C C C 转换为图像坐标系中的坐标 p p p ,这一过程可以用下面的等式表示:

p = C P p = CP p=CP

为了解这个方程,我们可以把它拆分为两部分,一部分对应 x x x 坐标,另一部分对应 y y y 坐标。对于二维点与三维点之间的对应关系,我们有:

x = ( c 11 X + c 12 Y + c 13 Z + c 14 ) / ( c 31 X + c 32 Y + c 33 Z + c 34 ) y = ( c 21 X + c 22 Y + c 23 Z + c 24 ) / ( c 31 X + c 32 Y + c 33 Z + c 34 ) \begin{align*} x &= (c_{11}X + c_{12}Y + c_{13}Z + c_{14}) / (c_{31}X + c_{32}Y + c_{33}Z + c_{34}) \\ y &= (c_{21}X + c_{22}Y + c_{23}Z + c_{24}) / (c_{31}X + c_{32}Y + c_{33}Z + c_{34}) \end{align*} xy=(c11X+c12Y+c13Z+c14)/(c31X+c32Y+c33Z+c34)=(c21X+c22Y+c23Z+c24)/(c31X+c32Y+c33Z+c34)

为了消除分母,我们可以将每个式子乘以分母的项,得到两个线性方程:

x ⋅ ( c 31 X + c 32 Y + c 33 Z + c 34 ) = c 11 X + c 12 Y + c 13 Z + c 14 y ⋅ ( c 31 X + c 32 Y + c 33 Z + c 34 ) = c 21 X + c 22 Y + c 23 Z + c 24 \begin{align*} x \cdot (c_{31}X + c_{32}Y + c_{33}Z + c_{34}) &= c_{11}X + c_{12}Y + c_{13}Z + c_{14} \\ y \cdot (c_{31}X + c_{32}Y + c_{33}Z + c_{34}) &= c_{21}X + c_{22}Y + c_{23}Z + c_{24} \end{align*} x(c31X+c32Y+c33Z+c34)y(c31X+c32Y+c33Z+c34)=c11X+c12Y+c13Z+c14=c21X+c22Y+c23Z+c24

当有两个摄像机和两个对应点时,我们可以将上面的方程组写为矩阵 A A A 的形式:

$$
Eigen::Vector3d TriangulatePoint(const Eigen::Matrix3x4d& cam1_from_world,
const Eigen::Matrix3x4d& cam2_from_world,
const Eigen::Vector2d& point1,
const Eigen::Vector2d& point2) {
Eigen::Matrix4d A;

A.row(0) = point1(0) * cam1_from_world.row(2) - cam1_from_world.row(0);
A.row(1) = point1(1) * cam1_from_world.row(2) - cam1_from_world.row(1);
A.row(2) = point2(0) * cam2_from_world.row(2) - cam2_from_world.row(0);
A.row(3) = point2(1) * cam2_from_world.row(2) - cam2_from_world.row(1);

Eigen::JacobiSVDEigen::Matrix4d svd(A, Eigen::ComputeFullV);

return svd.matrixV().col(3).hnormalized();
}
$$

3. 代码实现中:

A.row(0) = point1(0) * cam1_from_world.row(2) - cam1_from_world.row(0);
A.row(1) = point1(1) * cam1_from_world.row(2) - cam1_from_world.row(1);
A.row(2) = point2(0) * cam2_from_world.row(2) - cam2_from_world.row(0);
A.row(3) = point2(1) * cam2_from_world.row(2) - cam2_from_world.row(1);

这里的 point1(0)point1(1) 对应于上面方程中的 x x x y y ycam1_from_world.row(2) 是变换矩阵的第三行(我们称其为 c 3 i ( 1 ) c_{3i}^{(1)} c3i(1)),cam1_from_world.row(0)cam1_from_world.row(1) 分别对应变换矩阵的第一行和第二行(我们称其为 c 1 i ( 1 ) c_{1i}^{(1)} c1i(1) c 2 i ( 1 ) c_{2i}^{(1)} c2i(1))。对于第二个摄像机 cam2_from_world 也是同样的方法。

最后,通过计算 A A A 的奇异值分解(SVD)并且取 V V V 矩阵的最后一列,我们能够得到解空间的最小范数解。hnormalized() 函数是对坐标进行齐次归一化,从而得到三维空间中的点坐标。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

大江东去浪淘尽千古风流人物

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值