目录
2.4 遍历搜索搜索窗口中的所有潜在的匹配候选点,找到最优的和次优的
2.5 对最优次优结果进行检查,满足阈值、最优/次优比例,删除重复匹配
2.8 将最后通过筛选的匹配好的特征点保存到vbPrevMatched
用于单目初始化中用于参考帧和当前帧的特征点匹配。
- /**
- * @brief 单目初始化中用于参考帧和当前帧的特征点匹配
- * 步骤
- * Step 1 构建旋转直方图
- * Step 2 在半径窗口内搜索当前帧F2中所有的候选匹配特征点
- * Step 3 遍历搜索搜索窗口中的所有潜在的匹配候选点,找到最优的和次优的
- * Step 4 对最优次优结果进行检查,满足阈值、最优/次优比例,删除重复匹配
- * Step 5 计算匹配点旋转角度差所在的直方图
- * Step 6 筛除旋转直方图中“非主流”部分
- * Step 7 将最后通过筛选的匹配好的特征点保存
- * @param[in] F1 初始化参考帧
- * @param[in] F2 当前帧
- * @param[in & out] vbPrevMatched 本来存储的是参考帧的所有特征点坐标,该函数更新为匹配好的当前帧的特征点坐标
- * @param[in & out] vnMatches12 保存参考帧F1中特征点是否匹配上,index保存是F1对应特征点索引,值保存的是匹配好的F2特征点索引
- * @param[in] windowSize 搜索窗口
- * @return int 返回成功匹配的特征点数目
- */
- int ORBmatcher::SearchForInitialization(Frame &F1, Frame &F2, vector
&vbPrevMatched, vector<int> &vnMatches12, int windowSize) - {
- int nmatches=0;
- // F1中特征点和F2中匹配关系,注意是按照F1特征点数目分配空间
- vnMatches12 = vector<int>(F1.mvKeysUn.size(),-1);
-
- // Step 1 构建旋转直方图,HISTO_LENGTH = 30
- vector<int> rotHist[HISTO_LENGTH];
- for(int i=0;i
- // 每个bin里预分配500个,因为使用的是vector不够的话可以自动扩展容量
- rotHist[i].reserve(500);
-
- //! 原作者代码是 const float factor = 1.0f/HISTO_LENGTH; 是错误的,更改为下面代码
- const float factor = HISTO_LENGTH/360.0f;
-
- // 匹配点对距离,注意是按照F2特征点数目分配空间
- vector<int> vMatchedDistance(F2.mvKeysUn.size(),INT_MAX);
- // 从帧2到帧1的反向匹配,注意是按照F2特征点数目分配空间
- vector<int> vnMatches21(F2.mvKeysUn.size(),-1);
-
- // 遍历帧1中的所有特征点
- for(size_t i1=0, iend1=F1.mvKeysUn.size(); i1
- {
- cv::KeyPoint kp1 = F1.mvKeysUn[i1];
- int level1 = kp1.octave;
- // 只使用原始图像上提取的特征点
- if(level1>0)
- continue;
-
- // Step 2 在半径窗口内搜索当前帧F2中所有的候选匹配特征点
- // vbPrevMatched 输入的是参考帧 F1的特征点
- // windowSize = 100,输入最大最小金字塔层级 均为0
- vector<size_t> vIndices2 = F2.GetFeaturesInArea(vbPrevMatched[i1].x,vbPrevMatched[i1].y, windowSize,level1,level1);
-
- // 没有候选特征点,跳过
- if(vIndices2.empty())
- continue;
-
- // 取出参考帧F1中当前遍历特征点对应的描述子
- cv::Mat d1 = F1.mDescriptors.row(i1);
-
- int bestDist = INT_MAX; //最佳描述子匹配距离,越小越好
- int bestDist2 = INT_MAX; //次佳描述子匹配距离
- int bestIdx2 = -1; //最佳候选特征点在F2中的index
-
- // Step 3 遍历搜索搜索窗口中的所有潜在的匹配候选点,找到最优的和次优的
- for(vector<size_t>::iterator vit=vIndices2.begin(); vit!=vIndices2.end(); vit++)
- {
- size_t i2 = *vit;
- // 取出候选特征点对应的描述子
- cv::Mat d2 = F2.mDescriptors.row(i2);
- // 计算两个特征点描述子距离
- int dist = DescriptorDistance(d1,d2);
-
- if(vMatchedDistance[i2]<=dist)
- continue;
- // 如果当前匹配距离更小,更新最佳次佳距离
- if(dist
- {
- bestDist2=bestDist;
- bestDist=dist;
- bestIdx2=i2;
- }
- else if(dist
- {
- bestDist2=dist;
- }
- }
-
- // Step 4 对最优次优结果进行检查,满足阈值、最优/次优比例,删除重复匹配
- // 即使算出了最佳描述子匹配距离,也不一定保证配对成功。要小于设定阈值
- if(bestDist<=TH_LOW)
- {
- // 最佳距离比次佳距离要小于设定的比例,这样特征点辨识度更高
- if(bestDist<(float)bestDist2*mfNNratio)
- {
- // 如果找到的候选特征点对应F1中特征点已经匹配过了,说明发生了重复匹配,将原来的匹配也删掉
- if(vnMatches21[bestIdx2]>=0)
- {
- vnMatches12[vnMatches21[bestIdx2]]=-1;
- nmatches--;
- }
- // 次优的匹配关系,双向建立
- // vnMatches12保存参考帧F1和F2匹配关系,index保存是F1对应特征点索引,值保存的是匹配好的F2特征点索引
- vnMatches12[i1]=bestIdx2;
- vnMatches21[bestIdx2]=i1;
- vMatchedDistance[bestIdx2]=bestDist;
- nmatches++;
-
- // Step 5 计算匹配点旋转角度差所在的直方图
- if(mbCheckOrientation)
- {
- // 计算匹配特征点的角度差,这里单位是角度°,不是弧度
- float rot = F1.mvKeysUn[i1].angle-F2.mvKeysUn[bestIdx2].angle;
- if(rot<0.0)
- rot+=360.0f;
- // 前面factor = HISTO_LENGTH/360.0f
- // bin = rot / 360.of * HISTO_LENGTH 表示当前rot被分配在第几个直方图bin
- int bin = round(rot*factor);
- // 如果bin 满了又是一个轮回
- if(bin==HISTO_LENGTH)
- bin=0;
- assert(bin>=0 && bin
- rotHist[bin].push_back(i1);
- }
- }
- }
-
- }
-
- // Step 6 筛除旋转直方图中“非主流”部分
- if(mbCheckOrientation)
- {
- int ind1=-1;
- int ind2=-1;
- int ind3=-1;
- // 筛选出在旋转角度差落在在直方图区间内数量最多的前三个bin的索引
- ComputeThreeMaxima(rotHist,HISTO_LENGTH,ind1,ind2,ind3);
-
- for(int i=0; i
- {
- if(i==ind1 || i==ind2 || i==ind3)
- continue;
- // 剔除掉不在前三的匹配对,因为他们不符合“主流旋转方向”
- for(size_t j=0, jend=rotHist[i].size(); j
- {
- int idx1 = rotHist[i][j];
- if(vnMatches12[idx1]>=0)
- {
- vnMatches12[idx1]=-1;
- nmatches--;
- }
- }
- }
-
- }
-
- //Update prev matched
- // Step 7 将最后通过筛选的匹配好的特征点保存到vbPrevMatched
- for(size_t i1=0, iend1=vnMatches12.size(); i1
- if(vnMatches12[i1]>=0)
- vbPrevMatched[i1]=F2.mvKeysUn[vnMatches12[i1]].pt;
-
- return nmatches;
- }
2.1 参数解析
- /**
- * @brief 单目初始化中用于参考帧和当前帧的特征点匹配
- * 步骤
- * Step 1 构建旋转直方图
- * Step 2 在半径窗口内搜索当前帧F2中所有的候选匹配特征点
- * Step 3 遍历搜索搜索窗口中的所有潜在的匹配候选点,找到最优的和次优的
- * Step 4 对最优次优结果进行检查,满足阈值、最优/次优比例,删除重复匹配
- * Step 5 计算匹配点旋转角度差所在的直方图
- * Step 6 筛除旋转直方图中“非主流”部分
- * Step 7 将最后通过筛选的匹配好的特征点保存
- * @param[in] F1 初始化参考帧
- * @param[in] F2 当前帧
- * @param[in & out] vbPrevMatched 本来存储的是参考帧的所有特征点坐标,该函数更新为匹配好的当前帧的特征点坐标
- * @param[in & out] vnMatches12 保存参考帧F1中特征点是否匹配上,index保存是F1对应特征点索引,值保存的是匹配好的F2特征点索引
- * @param[in] windowSize 搜索窗口
- * @return int 返回成功匹配的特征点数目
- */
- int ORBmatcher::SearchForInitialization(Frame &F1, Frame &F2, vector
&vbPrevMatched, vector<int> &vnMatches12, int windowSize)
本函数的输入为两帧图片Frame &F1,Frame &F2及搜索窗口大小windowSize,输出为特征点的匹配关系vnMatches12、vbPrevMatched。
vnMatches12 = vector<int>(F1.mvKeysUn.size(),-1);
我们可以看到,vnMatches12是记录匹配关系的,大小定为帧1中经过去畸变后得到的特征点的数量。具体含义举个例子,比如vnMatches12[0] = 4的含义就是帧1中的第一个特征点和帧2中1的第5个特征点成功匹配。 刚开始我们全都初始化为-1说明匹配关系为空。
vbPrevMatched作为输出,记录的是匹配成功的点坐标信息。即如果第一帧的第五个特征点成功匹配到第二帧的第五十个特征点。vbPrevMatched[4]中储存的值为第二帧第五十个特征点的坐标。
用nmatches变量记录了已经匹配的特征点数目。
2.2 构建旋转直方图解析
- // Step 1 构建旋转直方图,HISTO_LENGTH = 30
- vector<int> rotHist[HISTO_LENGTH];
- for(int i=0;i
- // 每个bin里预分配500个,因为使用的是vector不够的话可以自动扩展容量
- rotHist[i].reserve(500);
-
- //! 原作者代码是 const float factor = 1.0f/HISTO_LENGTH; 是错误的,更改为下面代码
- const float factor = HISTO_LENGTH/360.0f;
-
- // 匹配点对距离,注意是按照F2特征点数目分配空间
- vector<int> vMatchedDistance(F2.mvKeysUn.size(),INT_MAX);
- // 从帧2到帧1的反向匹配,注意是按照F2特征点数目分配空间
- vector<int> vnMatches21(F2.mvKeysUn.size(),-1);
-
- // 遍历帧1中的所有特征点
- for(size_t i1=0, iend1=F1.mvKeysUn.size(); i1
- {
- cv::KeyPoint kp1 = F1.mvKeysUn[i1];
- int level1 = kp1.octave;
- // 只使用原始图像上提取的特征点
- if(level1>0)
- continue;
这段代码是在做什么呢?
挑选出在旋转角度差落在在直方图区间内数量最多的前三个直方图区间bin的索引,然后剔除掉不在前三的匹配对,因为他们不符合“主流旋转方向” 。
我们目标是建立如上的直方图,直方图分为两部分:每一个区间及每个区间所含的样本数量。
我们用vector rotHist[HISTO_LENGTH] 来构建了HISTO_LENGTH个区间,rotHist[i] 就是每个区间含的样本数量。(好奇为什么不用双层vector或者pair对组)。并初始化每个区间的样本数量初值为500。
匹配点对的汉明距离(描述子距离)vMatchedDistance,这里是按帧2初始化的,又定义了反向匹配vnMatches21,然后我们遍历第一帧的所有第一层特征点。
2.3 在半径窗口内搜索当前帧F2中所有的候选匹配特征点
- // Step 2 在半径窗口内搜索当前帧F2中所有的候选匹配特征点
- // vbPrevMatched 输入的是参考帧 F1的特征点
- // windowSize = 100,输入最大最小金字塔层级 均为0
- vector<size_t> vIndices2 = F2.GetFeaturesInArea(vbPrevMatched[i1].x,vbPrevMatched[i1].y, windowSize,level1,level1);
-
- // 没有候选特征点,跳过
- if(vIndices2.empty())
- continue;
-
- // 取出参考帧F1中当前遍历特征点对应的描述子
- cv::Mat d1 = F1.mDescriptors.row(i1);
-
- int bestDist = INT_MAX; //最佳描述子匹配距离,越小越好
- int bestDist2 = INT_MAX; //次佳描述子匹配距离
- int bestIdx2 = -1; //最佳候选特征点在F2中的index
参考链接:如何把去畸变的特征点分到栅格中
参考链接: GetFeaturesInArea函数解析
我们用vIndices2 变量接收所有帧一的正在迭代某个特征点的候选特征点kp1 匹配索引的集合,若vIndices2 为空,则帧2中没有与kp1匹配的候选特征点,这时需要对帧1的下一个特征点匹配候选特征点进行匹配。若帧2中有与kp1匹配的候选特征点,则取出参考帧F1中当前遍历特征点对应的描述子d1。
2.4 遍历搜索搜索窗口中的所有潜在的匹配候选点,找到最优的和次优的
- for(vector<size_t>::iterator vit=vIndices2.begin(); vit!=vIndices2.end(); vit++)
- {
- size_t i2 = *vit;
- // 取出候选特征点对应的描述子
- cv::Mat d2 = F2.mDescriptors.row(i2);
- // 计算两个特征点描述子距离
- int dist = DescriptorDistance(d1,d2);
-
- if(vMatchedDistance[i2]<=dist)
- continue;
- // 如果当前匹配距离更小,更新最佳次佳距离
- if(dist
- {
- bestDist2=bestDist;
- bestDist=dist;
- bestIdx2=i2;
- }
- else if(dist
- {
- bestDist2=dist;
- }
- }
取出每一个帧2中候选的特征点的描述子,计算帧一中特征点kp1与帧二中每个候选特征点的汉明距离,保存最佳和次加候选特征点的汉明距离和特征点索引。
2.5 对最优次优结果进行检查,满足阈值、最优/次优比例,删除重复匹配
2.6 计算匹配点旋转角度差所在的直方图
- if(bestDist<=TH_LOW)
- {
- // 最佳距离比次佳距离要小于设定的比例,这样特征点辨识度更高
- if(bestDist<(float)bestDist2*mfNNratio)
- {
- // 如果找到的候选特征点对应F1中特征点已经匹配过了,说明发生了重复匹配,将原来的匹配也删掉
- if(vnMatches21[bestIdx2]>=0)
- {
- vnMatches12[vnMatches21[bestIdx2]]=-1;
- nmatches--;
- }
- // 次优的匹配关系,双向建立
- // vnMatches12保存参考帧F1和F2匹配关系,index保存是F1对应特征点索引,值保存的是匹配好的F2特征点索引
- vnMatches12[i1]=bestIdx2;
- vnMatches21[bestIdx2]=i1;
- vMatchedDistance[bestIdx2]=bestDist;
- nmatches++;
-
- // Step 5 计算匹配点旋转角度差所在的直方图
- if(mbCheckOrientation)
- {
- // 计算匹配特征点的角度差,这里单位是角度°,不是弧度
- float rot = F1.mvKeysUn[i1].angle-F2.mvKeysUn[bestIdx2].angle;
- if(rot<0.0)
- rot+=360.0f;
- // 前面factor = HISTO_LENGTH/360.0f
- // bin = rot / 360.of * HISTO_LENGTH 表示当前rot被分配在第几个直方图bin
- int bin = round(rot*factor);
- // 如果bin 满了又是一个轮回
- if(bin==HISTO_LENGTH)
- bin=0;
- assert(bin>=0 && bin
- rotHist[bin].push_back(i1);
- }
- }
- }
- // 要用到的一些阈值
- const int ORBmatcher::TH_HIGH = 100;
- const int ORBmatcher::TH_LOW = 50;
- const int ORBmatcher::HISTO_LENGTH = 30;
并不是上一步中得到的由汉明距离匹配的特征点就是最终匹配的特征点,还要满足以下条件:
如果最佳匹配点的汉明距离大于TH_LOW,则匹配失败。(汉明距离大意味着匹配精度低)
如果最佳匹配点的汉明距离/次佳匹配点的汉明距离大于一定的比例,则匹配失败(保证特征点的辨识度)
如果候选的特征点已经和帧一之前的某个特征点匹配过了,则匹配失败(两次一定有一次是误匹配)
如果满足这三点,则”暂时“认为特征点匹配成功,更新匹配容器vnMatches12、vnMatches21、vMatchedDistance表明匹配成功,并将成功匹配特征点的数量nmatches+1。
bool mbCheckOrientation; ///< 是否检查特征点的方向
若此标志位为true,我们还要检查特征点的方向构建旋转方向直方图再剔除一部分匹配好的特征点作为最终匹配的特征点输出给上层调用。
关于特征点的angle属性如何计算及意义是什么,我在之前的博客有说,放到了下面链接中。
我们计算两个特征点的angle差rot,使其在0°-360°之间。
我们前面定义了factor = HISTO_LENGTH/360.0f,这里HISTO_LENGTH是分布直方图的区间数目,这里factor * rot = 角度/360° * 分布直方图的区间数目,即匹配的特征点的角度差落在哪个分布直方图的区间内,然后将帧2中匹配的特征点放入直方图的一个区间内。
举一个例子吧:
如果计算角度差值为5° 则5/360 * 30 = 0.42 放入第0个直方图内。
如果计算角度差值为15° 则15/360 * 30 = 1.26 放入第1个直方图内。
如果计算角度差值为140° 则140/360 * 30 = 11.67 放入第11个直方图内。ORB-SLAM2 ---- IC_Angle函数_Courage2022的博客-CSDN博客https://blog.csdn.net/qq_41694024/article/details/126306830
2.7 筛除旋转直方图中“非主流”部分
- if(mbCheckOrientation)
- {
- int ind1=-1;
- int ind2=-1;
- int ind3=-1;
- // 筛选出在旋转角度差落在在直方图区间内数量最多的前三个bin的索引
- ComputeThreeMaxima(rotHist,HISTO_LENGTH,ind1,ind2,ind3);
-
- for(int i=0; i
- {
- if(i==ind1 || i==ind2 || i==ind3)
- continue;
- // 剔除掉不在前三的匹配对,因为他们不符合“主流旋转方向”
- for(size_t j=0, jend=rotHist[i].size(); j
- {
- int idx1 = rotHist[i][j];
- if(vnMatches12[idx1]>=0)
- {
- vnMatches12[idx1]=-1;
- nmatches--;
- }
- }
- }
-
- }
bool mbCheckOrientation; ///< 是否检查特征点的方向
若此标志位为true,我们还要检查特征点的方向构建旋转方向直方图再剔除一部分匹配好的特征点作为最终匹配的特征点输出给上层调用。
这块就是挑选出在旋转角度差落在在直方图区间内数量最多的前三个直方图区间bin的索引,然后剔除掉不在前三的匹配对,因为他们不符合“主流旋转方向” 。
这是因为两张图片各个特征点在计算angle属性时各个特征点都是要计算这个旋转到x轴的角度的,那么它们的差值肯定趋于一致,若差别较大,则考虑特征点误匹配的几率大大提高。
2.8 将最后通过筛选的匹配好的特征点保存到vbPrevMatched
- for(size_t i1=0, iend1=vnMatches12.size(); i1
- if(vnMatches12[i1]>=0)
- vbPrevMatched[i1]=F2.mvKeysUn[vnMatches12[i1]].pt;
-
- return nmatches;
若帧1到帧2中帧1的某个特征点成功匹配到帧2的某个特征点,则更新vbPrevMatched容器的成功匹配帧一到帧二特征点的点坐标信息。
即如果第一帧的第五个特征点成功匹配到第二帧的第五十个特征点。
vbPrevMatched[4]中储存的值为第二帧第五十个特征点的坐标。
最后返回匹配成功的特征点对数给上层函数
-
相关阅读:
VSC/SMC(十六)——自适应鲁棒滑模控制
在线代码编辑器CodePen和CodeSandbox
python使用flask实现前后端分离&通过前端修改数据库数据【全栈开发基础】
ChatGPT的强大之处:探究及与国内产品的对比
xss攻击与csrf攻击
微信小程序 24 播放音乐页的完善①
【Nowcoder-TOP101】BM1.小明打联盟
内容合规管理|内容合规审核难?帷幄AI审核员上线!
Autojs 利用OpenCV识别棋子之天天象棋你马没了
下一代 SCA:流水线成分分析
-
原文地址:https://blog.csdn.net/qq_41694024/article/details/126319167