若该文为原创文章,转载请注明原文出处
本文章博客地址:https://hpzwl.blog.csdn.net/article/details/143432922
长沙红胖子Qt(长沙创微智科)博文大全:开发技术集合(包含Qt实用技术、树莓派、三维、OpenCV、OpenGL、ffmpeg、OSG、单片机、软硬结合等等)持续更新中...
OpenCV开发专栏(点击传送门)
上一篇:《OpenCV开发笔记(八十一):通过棋盘格使用鱼眼方式标定相机内参矩阵矫正摄像头图像》
下一篇:持续补充中...
前言
对于图像拼接,前面探讨了通过基于Stitcher进行拼接过渡和基于特征点进行拼接过渡,这2个过渡的方式是摄像头拍摄角度和方向不应差距太大。
对于特定的场景,本身摄像头拍摄角度差距较大,拉伸变换后也难做到完美的缝隙拼接,这个时候使用渐近过渡反倒是最好的。
Demo
单独蒙版
data:image/s3,"s3://crabby-images/afb18/afb18c45960fb52f1d6b54bf49dcc1f486567fc1" alt=""
data:image/s3,"s3://crabby-images/3eaaf/3eaaf7bf2b343f1b309a368300bf4c868b43822f" alt=""
data:image/s3,"s3://crabby-images/32c96/32c9639b5001dad2cf1391b4f3ea7b3eb74452a1" alt=""
data:image/s3,"s3://crabby-images/e6d32/e6d325bbc499388b45a86cc59dbb58cd184aec52" alt=""
蒙版过渡,这里只是根据图来,其实可对每个像素对于第一张图为系数k,而第二张为255-k,实现渐近过渡。
data:image/s3,"s3://crabby-images/4dba2/4dba267d26c48ddd2d1eeef2a5f20c6d424521c0" alt=""
data:image/s3,"s3://crabby-images/b590c/b590cdfd9af2e983a9ef9351341987f959b15a95" alt=""
data:image/s3,"s3://crabby-images/9d1ca/9d1caa48d247875d115d3b508e78b6d5b8edbf2a" alt=""
data:image/s3,"s3://crabby-images/8cbbb/8cbbb9cc8b51608bbe1ef7c9e902c4495accf3ea" alt=""
直接使用第一张蒙版优化
data:image/s3,"s3://crabby-images/6dde9/6dde9811001dd6929d3414c52e3a561a7d993f28" alt=""
data:image/s3,"s3://crabby-images/0742f/0742fe041be4d753da65fe91094d011a04fc52c3" alt=""
data:image/s3,"s3://crabby-images/067ce/067ce6d1db0c87a3546220348d3857cc49bfa9e5" alt=""
准本蒙版
蒙版可以混合,也可以分开,为了让读者更好的深入理解原理,这里都使用:
找个工具,造单色渐进色,红色蒙版,只是r通道,bga都为0
data:image/s3,"s3://crabby-images/ec442/ec442364c5120059f68b75b47b88eec9fdb08cdc" alt=""
(注意:使用rgba四通道)
data:image/s3,"s3://crabby-images/238d5/238d5bbbc71f34e5e0a52f5e47907932429f08a3" alt=""
(上面这张图,加了边框,导致了"入坑二"打印像素值不对)
data:image/s3,"s3://crabby-images/29b0d/29b0d6071266734309f527712f41f8aada2dea20" alt=""
由于工具渐进色无法叠层,这个工具无法实现rgba不同向渐进色再一张图(横向、纵向、斜向),更改了方式,每个使用一张图:
为了方便,不管a通道了,直接a为100%(255)。
data:image/s3,"s3://crabby-images/7ce5c/7ce5c5100a8704bde6f297c87d58217e53292756" alt=""
再弄另外一个通道的:
data:image/s3,"s3://crabby-images/5f885/5f8858736c1599a13fca57cd7ab9520f6b2ff9b4" alt=""
在这里使用工具就只能单独一张了:
data:image/s3,"s3://crabby-images/0de26/0de26faa10444e51ae9e17c74ee456b20be76504" alt=""
一个蒙版图的过渡实例
步骤一:打开图片和蒙版
data:image/s3,"s3://crabby-images/b6dd9/b6dd9dca66dbe49e2c0fcf05922037b80d179625" alt=""
cpp
cv::Mat matLeft = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/29.jpg");
cv::Mat matRight = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/30.jpg");
cv::Mat matMask1 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/37.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask2 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/38.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask3 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/39.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask4 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/40.png", cv::IMREAD_UNCHANGED);
步骤二:将蒙版变成和原图一样大小
data:image/s3,"s3://crabby-images/023fb/023fb898b4ffaf7a6a0d9f6e8771b2bdbc285788" alt=""
cpp
cv::resize(matLeft, matLeft, cv::Size(0, 0), 0.5, 0.5);
cv::resize(matRight, matRight, cv::Size(0, 0), 0.5, 0.5);
cv::resize(matMask1, matMask1, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask2, matMask2, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask3, matMask3, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask4, matMask4, cv::Size(matLeft.cols, matLeft.rows));
步骤三:底图
由于两张图虽然是同样大小,但是其不是按照整体拼接后的大小,所以需要假设一个拼接后的大小的底图。
data:image/s3,"s3://crabby-images/50551/5055162fce9ad986c5cc9ba20d5e8d5fe2802ca1" alt=""
cpp
// 底图,扩大500横向,方便移动
cv::Mat matResult = cv::Mat(matLeft.rows, matLeft.cols + 500, CV_8UC3);
步骤四:原图融合
data:image/s3,"s3://crabby-images/2ae0a/2ae0a059893df921a62e6f3d05b665e9caab3f13" alt=""
cpp
// 副本,每次都要重新清空来调整
cv::Mat matResult2 = matResult.clone();
#if 1
// 第一张图,直接比例赋值,因为底图为0
for(int row = 0; row < matLeft.rows; row++)
{
for(int col = 0; col < matLeft.cols; col++)
{
double r = matMask1.at<cv::Vec4b>(row, col)[2] / 255.0f;
// double r = matMask2.at<cv::Vec4b>(row, col)[1] / 255.0f;
// double r = matMask3.at<cv::Vec4b>(row, col)[0] / 255.0f;
// double r = matMask4.at<cv::Vec4b>(row, col)[0] / 255.0f;
matResult2.at<cv::Vec3b>(row, col)[0] = (matLeft.at<cv::Vec3b>(row, col)[0] * r);
matResult2.at<cv::Vec3b>(row, col)[1] = (matLeft.at<cv::Vec3b>(row, col)[1] * r);
matResult2.at<cv::Vec3b>(row, col)[2] = (uchar)(matLeft.at<cv::Vec3b>(row, col)[2] * r);
}
}
#endif
步骤五:另外一张图的融合
data:image/s3,"s3://crabby-images/cc0d1/cc0d154079156ef7a0d71dce93e616cda7ab5209" alt=""
cpp
#if 1
// 第二张图,加法,因为底图为原图了
for(int row = 0; row < matRight.rows; row++)
{
for(int col = 0; col < matRight.cols; col++)
{
double g = matMask2.at<cv::Vec4b>(row, col)[1] / 255.0f;
// 偏移了x坐标
matResult2.at<cv::Vec3b>(row, col + x)[0] += matRight.at<cv::Vec3b>(row, col)[0] * g;
matResult2.at<cv::Vec3b>(row, col + x)[1] += matRight.at<cv::Vec3b>(row, col)[1] * g;
matResult2.at<cv::Vec3b>(row, col + x)[2] += matRight.at<cv::Vec3b>(row, col)[2] * g;
}
}
#endif
步骤六(与步骤五互斥):优化的融合
data:image/s3,"s3://crabby-images/faa3a/faa3a9f3fee89a6d4dfc64c27089ea0b686aa74c" alt=""
cpp
#if 1
// 第二张图,加法,因为底图为原图了(优化)
for(int row = 0; row < matRight.rows; row++)
{
for(int col = 0; col < matRight.cols; col++)
{
double r2;
if(x + col <= matLeft.cols)
{
r2 = (255 - matMask1.at<cv::Vec4b>(row, col + x)[2]) / 255.0f;
}else{
r2 = 1.0f;
}
// 偏移了x坐标
matResult2.at<cv::Vec3b>(row, col + x)[0] += matRight.at<cv::Vec3b>(row, col)[0] * r2;
matResult2.at<cv::Vec3b>(row, col + x)[1] += matRight.at<cv::Vec3b>(row, col)[1] * r2;
matResult2.at<cv::Vec3b>(row, col + x)[2] += matRight.at<cv::Vec3b>(row, col)[2] * r2;
}
}
#endif
函数原型
手码的像素算法,没有什么高级函数。
Demo源码
cpp
void OpenCVManager::testMaskSplicing()
{
cv::Mat matLeft = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/29.jpg");
cv::Mat matRight = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/30.jpg");
cv::Mat matMask1 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/37.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask2 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/38.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask3 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/39.png", cv::IMREAD_UNCHANGED);
cv::Mat matMask4 = cv::imread("D:/qtProject/openCVDemo/openCVDemo/modules/openCVManager/images/40.png", cv::IMREAD_UNCHANGED);
#if 0
// 打印通道数和数据类型
// ..\openCVDemo\modules\openCVManager\OpenCVManager.cpp 9166 "2024-10-31 20:07:42:619" 4 24 24
LOG << matMask.channels() << matMask.type() << CV_8UC4; // 4 24
// 打印mask蒙版行像素,隔一定行数打一次
for(int row = 0; row < matMask.rows; row += 10)
{
for(int col = 100; col < matMask.cols; col++)
{
int r = matMask.at<cv::Vec4b>(row, col)[2];
int g = matMask.at<cv::Vec4b>(row, col)[1];
int b = matMask.at<cv::Vec4b>(row, col)[0];
int a = matMask.at<cv::Vec4b>(row, col)[3];
LOG << "row:" << row << ", col:" << col << "r(rgba):" << r << g << b << a;
break;
}
}
#endif
// 图片较大,缩为原来的0.5倍
cv::resize(matLeft, matLeft, cv::Size(0, 0), 0.5, 0.5);
cv::resize(matRight, matRight, cv::Size(0, 0), 0.5, 0.5);
cv::resize(matMask1, matMask1, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask2, matMask2, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask3, matMask3, cv::Size(matLeft.cols, matLeft.rows));
cv::resize(matMask4, matMask4, cv::Size(matLeft.cols, matLeft.rows));
// 底图,扩大500横向,方便移动
cv::Mat matResult = cv::Mat(matLeft.rows, matLeft.cols + 500, CV_8UC3);
// 第一张图
int key = 0;
int x = 0;
while(true)
{
// 副本,每次都要重新清空来调整
cv::Mat matResult2 = matResult.clone();
#if 1
// 第一张图,直接比例赋值,因为底图为0
for(int row = 0; row < matLeft.rows; row++)
{
for(int col = 0; col < matLeft.cols; col++)
{
double r = matMask1.at<cv::Vec4b>(row, col)[2] / 255.0f;
// double r = matMask2.at<cv::Vec4b>(row, col)[1] / 255.0f;
// double r = matMask3.at<cv::Vec4b>(row, col)[0] / 255.0f;
// double r = matMask4.at<cv::Vec4b>(row, col)[0] / 255.0f;
matResult2.at<cv::Vec3b>(row, col)[0] = (matLeft.at<cv::Vec3b>(row, col)[0] * r);
matResult2.at<cv::Vec3b>(row, col)[1] = (matLeft.at<cv::Vec3b>(row, col)[1] * r);
matResult2.at<cv::Vec3b>(row, col)[2] = (uchar)(matLeft.at<cv::Vec3b>(row, col)[2] * r);
}
}
#endif
#if 0
// 第二张图,加法,因为底图为原图了
for(int row = 0; row < matRight.rows; row++)
{
for(int col = 0; col < matRight.cols; col++)
{
double g = matMask2.at<cv::Vec4b>(row, col)[1] / 255.0f;
// 偏移了x坐标
matResult2.at<cv::Vec3b>(row, col + x)[0] += matRight.at<cv::Vec3b>(row, col)[0] * g;
matResult2.at<cv::Vec3b>(row, col + x)[1] += matRight.at<cv::Vec3b>(row, col)[1] * g;
matResult2.at<cv::Vec3b>(row, col + x)[2] += matRight.at<cv::Vec3b>(row, col)[2] * g;
}
}
#endif
#if 1
// 第二张图,加法,因为底图为原图了(优化)
for(int row = 0; row < matRight.rows; row++)
{
for(int col = 0; col < matRight.cols; col++)
{
double r2;
if(x + col <= matLeft.cols)
{
r2 = (255 - matMask1.at<cv::Vec4b>(row, col + x)[2]) / 255.0f;
}else{
r2 = 1.0f;
}
// 偏移了x坐标
matResult2.at<cv::Vec3b>(row, col + x)[0] += matRight.at<cv::Vec3b>(row, col)[0] * r2;
matResult2.at<cv::Vec3b>(row, col + x)[1] += matRight.at<cv::Vec3b>(row, col)[1] * r2;
matResult2.at<cv::Vec3b>(row, col + x)[2] += matRight.at<cv::Vec3b>(row, col)[2] * r2;
}
}
#endif
// cv::imshow("matMask1", matMask1);
// cv::imshow("matLeft", matLeft);
cv::imshow("matResult2", matResult2);
key = cv::waitKey(0);
if(key == 'a')
{
x--;
if(x < 0)
{
x = 0;
}
}else if(key == 'd')
{
x++;
if(x + matRight.cols > matResult2.cols)
{
x = matResult2.cols - matRight.cols;
}
}else if(key == 'q')
{
break;
}
}
}
工程模板v1.72.0
data:image/s3,"s3://crabby-images/fa8c2/fa8c279fc7865c74726986612aabc69c596e9986" alt=""
入坑
入坑一:读取通道rgba失败
问题:读取通道rgba失败
data:image/s3,"s3://crabby-images/bf8ba/bf8baf72ee2078a6e753ad11328b1f9d69c3ff4e" alt=""
原因
是uchar,转换成byte,而不是int
data:image/s3,"s3://crabby-images/61d54/61d5475cb800dec8ea29e184d4df250ff780a741" alt=""
解决
data:image/s3,"s3://crabby-images/5c34c/5c34caa80e59adc032a448ae8013e47c4b6d608c" alt=""
data:image/s3,"s3://crabby-images/a95f5/a95f5296f1f2785d0152803371c7415e4904a83b" alt=""
入坑二:读取通道一直是0,0,0,255
问题
读取通道一直是0,0,0,255。
data:image/s3,"s3://crabby-images/e4467/e44671feabab75264eb7ecd6102779f51692d39d" alt=""
原因
弄了张图,还是255,然后发现是为了截图更清楚,弄了个边框,而我们打印正好是打印了0位置。
data:image/s3,"s3://crabby-images/5495a/5495a9ed665a47a236a041b82ae1ed22a70f6adb" alt=""
data:image/s3,"s3://crabby-images/af400/af400c27d39217aa7d9e0182b524f4710ba1bb87" alt=""
解决
最终是要去掉边框,没边框就是空看不出,如下图:
data:image/s3,"s3://crabby-images/91e9b/91e9bb5b57d965b1a7c964636c1ced285d93a8c8" alt=""
data:image/s3,"s3://crabby-images/8b4cf/8b4cf38f765c0cac59f3d4496014fa5959b46557" alt=""
入坑三:过渡有黑线赋值不对
问题
直接位置赋值,出现条纹
data:image/s3,"s3://crabby-images/10ee0/10ee0c9ad29e01f5743c1bf38e97f518246ee93b" alt=""
data:image/s3,"s3://crabby-images/1800d/1800d886a831452bb95061ef66647689dad68183" alt=""
原因
类型是vec4b
data:image/s3,"s3://crabby-images/04aa0/04aa0814812923513b2be9d6bdccc014462fa5c8" alt=""
解决
data:image/s3,"s3://crabby-images/a4816/a48163deadea0afaf56f3db299b08fc6d945127c" alt=""
data:image/s3,"s3://crabby-images/93594/9359437d796168afeb7f339453ad47f1cff521a6" alt=""
入坑四:原图融合比例有黑线
问题
data:image/s3,"s3://crabby-images/27a32/27a32db57c04d1bc9e41140e8207c522f3f640af" alt=""
原因
跟上面一样,mask蒙版是rgba的,需要vec4b
data:image/s3,"s3://crabby-images/16743/16743f7018af752d41040f643f684f7575d16792" alt=""
解决
data:image/s3,"s3://crabby-images/618b0/618b0cdf54ae3dd1e63a68803513f6d346adb57b" alt=""
data:image/s3,"s3://crabby-images/c2793/c27939137012b09f926f97f7450f14f8cf80e09a" alt=""
上一篇:《OpenCV开发笔记(八十一):通过棋盘格使用鱼眼方式标定相机内参矩阵矫正摄像头图像》
下一篇:持续补充中...
本文章博客地址:https://hpzwl.blog.csdn.net/article/details/143432922