OpenCV 4.11.0
开源计算机视觉库
加载中...
搜索中...
未找到匹配项
samples/cpp/distrans.cpp

一个使用距离变换的例子

#include <stdio.h>
using namespace std;
using namespace cv;
int maskSize0 = DIST_MASK_5;
int voronoiType = -1;
int edgeThresh = 100;
int distType0 = DIST_L1;
// 输出和临时图像
Mat gray;
// 阈值跟踪条回调函数
static void onTrackbar( int, void* )
{
static const Scalar colors[] =
{
Scalar(0,0,0),
Scalar(255,0,0),
Scalar(255,128,0),
Scalar(255,255,0),
Scalar(0,255,0),
Scalar(0,128,255),
Scalar(0,255,255),
Scalar(0,0,255),
Scalar(255,0,255)
};
int maskSize = voronoiType >= 0 ? DIST_MASK_5 : maskSize0;
int distType = voronoiType >= 0 ? DIST_L2 : distType0;
Mat edge = gray >= edgeThresh, dist, labels, dist8u;
if( voronoiType < 0 )
distanceTransform( edge, dist, distType, maskSize );
else
distanceTransform( edge, dist, labels, distType, maskSize, voronoiType );
if( voronoiType < 0 )
{
// 开始“绘制”距离变换结果
dist *= 5000;
pow(dist, 0.5, dist);
Mat dist32s, dist8u1, dist8u2;
dist.convertTo(dist32s, CV_32S, 1, 0.5);
dist32s &= Scalar::all(255);
dist32s.convertTo(dist8u1, CV_8U, 1, 0);
dist32s *= -1;
dist32s += Scalar::all(255);
dist32s.convertTo(dist8u2, CV_8U);
Mat planes[] = {dist8u1, dist8u2, dist8u2};
merge(planes, 3, dist8u);
}
else
{
dist8u.create(labels.size(), CV_8UC3);
for( int i = 0; i < labels.rows; i++ )
{
const int* ll = (const int*)labels.ptr(i);
const float* dd = (const float*)dist.ptr(i);
uchar* d = (uchar*)dist8u.ptr(i);
for( int j = 0; j < labels.cols; j++ )
{
int idx = ll[j] == 0 || dd[j] == 0 ? 0 : (ll[j]-1)%8 + 1;
float scale = 1.f/(1 + dd[j]*dd[j]*0.0004f);
int b = cvRound(colors[idx][0]*scale);
int g = cvRound(colors[idx][1]*scale);
int r = cvRound(colors[idx][2]*scale);
d[j*3] = (uchar)b;
d[j*3+1] = (uchar)g;
d[j*3+2] = (uchar)r;
}
}
}
imshow("距离图", dist8u );
}
static void help(const char** argv)
{
printf("\n演示在边缘图像之间使用距离变换函数的程序。\n"
"用法:\n"
"%s [图像名称 -- 默认图像是 stuff.jpg]\n"
"\n热键:\n"
"\tESC - 退出程序\n"
"\tC - 使用 C/Inf 度量\n"
"\tL1 - 使用 L1 度量\n"
"\tL2 - 使用 L2 度量\n"
"\t3 - 使用 3x3 掩码\n"
"\t5 - 使用 5x5 掩码\n"
"\t0 - 使用精确距离变换\n"
"\tv - 切换到 Voronoi 图模式\n"
"\tp - 切换到基于像素的 Voronoi 图模式\n"
"\tSPACE - 循环遍历所有模式\n\n", argv[0]);
}
const char* keys =
{
"{help h||}{@image |stuff.jpg|输入图像文件}"
};
int main( int argc, const char** argv )
{
CommandLineParser parser(argc, argv, keys);
help(argv);
if (parser.has("help"))
return 0;
string filename = parser.get<string>(0);
gray = imread(samples::findFile(filename), 0);
if(gray.empty())
{
printf("无法读取图像文件:%s\n", filename.c_str());
help(argv);
return -1;
}
namedWindow("距离图", 1);
createTrackbar("亮度阈值", "距离图", &edgeThresh, 255, onTrackbar, 0);
for(;;)
{
// 更新视图的调用
onTrackbar(0, 0);
char c = (char)waitKey(0);
if( c == 27 )
break;
if( c == 'c' || c == 'C' || c == '1' || c == '2' ||
c == '3' || c == '5' || c == '0' )
voronoiType = -1;
if( c == 'c' || c == 'C' )
distType0 = DIST_C;
else if( c == '1' )
distType0 = DIST_L1;
else if( c == '2' )
distType0 = DIST_L2;
else if( c == '3' )
maskSize0 = DIST_MASK_3;
else if( c == '5' )
maskSize0 = DIST_MASK_5;
否则如果( c == '0' )
maskSize0 = DIST_MASK_PRECISE;
否则如果( c == 'v' )
voronoiType = 0;
否则如果( c == 'p' )
voronoiType = 1;
否则如果( c == ' ' )
{
如果( voronoiType == 0 )
voronoiType = 1;
否则如果( voronoiType == 1 )
{
voronoiType = -1;
maskSize0 = DIST_MASK_3;
distType0 = DIST_C;
}
否则如果( distType0 == DIST_C )
distType0 = DIST_L1;
否则如果( distType0 == DIST_L1 )
distType0 = DIST_L2;
否则如果( maskSize0 == DIST_MASK_3 )
maskSize0 = DIST_MASK_5;
否则如果( maskSize0 == DIST_MASK_5 )
maskSize0 = DIST_MASK_PRECISE;
否则如果( maskSize0 == DIST_MASK_PRECISE )
voronoiType = 0;
}
}
return 0;
}
用于命令行解析。
定义 utility.hpp:890
n维稠密数组类
定义 mat.hpp:829
void convertTo(OutputArray m, int rtype, double alpha=1, double beta=0) const
将数组转换为另一种数据类型,并可以选择缩放。
#define CV_8U
定义 interface.h:73
#define CV_32S
定义 interface.h:77
unsigned char uchar
定义 interface.h:51
#define CV_8UC3
定义 interface.h:90
int cvRound(double value)
将浮点数四舍五入到最接近的整数。
定义 fast_math.hpp:200
int waitKey(int delay=0)
等待按下按键。
void namedWindow(const String &winname, int flags=WINDOW_AUTOSIZE)
创建一个窗口。
int createTrackbar(const String &trackbarname, const String &winname, int *value, int count, TrackbarCallback onChange=0, void *userdata=0)
创建一个轨迹条并将其附加到指定的窗口。
CV_EXPORTS_W Mat imread(const String &filename, int flags=IMREAD_COLOR_BGR)
从文件中加载图像。
@ DIST_C
距离 = max(|x1-x2|,|y1-y2|)
定义 imgproc.hpp:308
@ DIST_L1
距离 = |x1-x2| + |y1-y2|
定义 imgproc.hpp:306
@ DIST_L2
简单的欧几里得距离
定义 imgproc.hpp:307
@ DIST_MASK_3
mask=3
定义 imgproc.hpp:317
@ DIST_MASK_PRECISE
定义 imgproc.hpp:319
@ DIST_MASK_5
mask=5
定义 imgproc.hpp:318
int main(int argc, char *argv[])
定义 highgui_qt.cpp:3
定义 core.hpp:107
STL 命名空间。