很久之前学习过一段时间的OpenCV,当时没有做什么笔记,但是代码都还在,这里把它贴出来做个记录。

代码放在码云上,地址在这里https://gitee.com/solym/OpenCVTest/tree/master/OpenCVTest

效果

代码

#include <opencv2/core.hpp>
#include <opencv2/highgui.hpp>
#include <opencv2/imgproc.hpp>
#include <opencv2/legacy/legacy.hpp> // cvSnakeImage // http://blog.csdn.net/hongxingabc/article/details/51606520
//http://wiki.opencv.org.cn/index.php/Snake%E8%BD%AE%E5%BB%93%E4%BE%8B%E5%AD%90 static cv::Mat src, src_bak; // 原始图像及其备份 static int thresholdness = 128; // 阈值
// 下面的参数用于滑块的值传出
static int ialpha = 20; // alpha代表点相互靠拢的权值(0-1.0)
static int ibeta = 20; // beta表示弯曲能量(越小越容易弯曲)(0-1.0)
static int igamma = 20; // gamma表示整体能量(0-1.0) void on_change(int pos,void*); int snakeContour()
{
// 创建窗口
cv::namedWindow("win", 0);
// 添加四个进度条(滑块条)
cv::createTrackbar("Thd", "win", &thresholdness, 255, on_change);
cv::createTrackbar("alpha", "win", &ialpha, 100, on_change);
cv::createTrackbar("beta", "win", &ibeta, 100, on_change);
cv::createTrackbar("gamma", "win", &igamma, 100, on_change); cv::resizeWindow("win", 640, 480);
// 调用一次on_change,以便显示出窗口
on_change(0,NULL); while (true){
if (cv::waitKey(40) == 27) {
cv::destroyWindow("win");
break; // 按下ESC键就退出
}
// 显示图像到窗口
cv::imshow("win", src_bak);
} return 0;
} void on_change(int,void*)
{
// 读取原图像
src_bak = cv::imread("../Image/sisy.jpg", 1);
// 转换为灰度图像
cv::cvtColor(src_bak, src, CV_BGR2GRAY); // 对单通道数组应用固定阈值操作。该函数的典型应用是对灰度图像进行阈值操作得到二值图像。
// 二值化操作
cv::threshold(src/*原始图像(单通道)*/,
src/*目标图像(与原始图像类型一致)*/,
thresholdness/*二值化控制阈值*/,
255/*使用 CV_THRESH_BINARY 和 CV_THRESH_BINARY_INV 的最大值*/,
CV_THRESH_BINARY/*阈值类型*/);
// CV_THRESH_BINARY:如果 src(x, y)>threshold, dst(x, y) = max_value; 否则, dst(x, y) = 0;
// CV_THRESH_BINARY_INV:如果 src(x, y)>threshold, dst(x, y) = 0; 否则, dst(x, y) = max_value.
// CV_THRESH_TRUNC:如果 src(x, y)>threshold,dst(x, y) = max_value; 否则dst(x, y) = src(x, y).
// CV_THRESH_TOZERO:如果src(x, y)>threshold,dst(x, y) = src(x, y); 否则 dst(x, y) = 0。
// CV_THRESH_TOZERO_INV:如果 src(x, y)>threshold,dst(x, y) = 0; 否则dst(x, y) = src(x, y). std::vector<std::vector<cv::Point> > contours; // http://www.360doc.com/content/12/1028/14/6408986_244247418.shtml
// 寻找初始化轮廓
cv::findContours(src/*输入图像(2值单通道图像)*/,
contours/*传出检测的轮廓点数组*/,
CV_RETR_EXTERNAL/*轮廓检测模式*/,
CV_CHAIN_APPROX_SIMPLE/*轮廓近似方式*/,
cv::Point(0, 0)/*轮廓点偏移量*/); if (contours.empty()) {
return; //没有检测到轮廓
} // 绘制轮廓线(绘制到备份的图像上)
cv::drawContours(src_bak, contours, -1, cv::Scalar(0, 0, 255), 2, 8); {
// 获取alpha、beta、gamma三个值
float alpha = ialpha / 100.0f;
float beta = ibeta / 100.0f;
float gamma = igamma / 100.0f; // 改变轮廓位置使得它的能量最小
// 因为没有C++的接口,所以这里使用C的接口
IplImage ipl_src = IplImage(src); // 每个点用于搜索最小值的邻域尺寸,两个 win.width 和 win.height 都必须是奇数
CvSize size;
size.width = 3; size.height = 3;
// 终止条件
CvTermCriteria criteria;
criteria.type = CV_TERMCRIT_ITER;
criteria.max_iter = 1000;
criteria.epsilon = 0.1;
// 梯度符号。如果非零,函数为每一个图像象素计算梯度幅值,且把它当成能量场,否则考虑输入图像本身。
int calc_gradient = 1; std::vector<CvPoint> point_arr; // 用于获取一个轮廓环的特征点
for (int nline = 0; nline < contours.size(); ++nline) {
// 当前轮廓线点数
int point_count = contours[nline].size();
if (point_count > 2) { // 低于3个点的时候,会出错
point_arr.resize(point_count);
} else {
continue;
}
// 拷贝特征点
for (int npt = 0; npt < contours[nline].size(); ++npt) {
point_arr[npt] = contours[nline][npt];
}
// 执行改变轮廓位置使得它的能量最小
cvSnakeImage(&ipl_src/*输入图像或外部能量域*/,
point_arr.data()/*传入传出轮廓点*/,
point_count/*传入轮廓点数组大小*/,
&alpha/*连续性能量的权 Weight[s],单个浮点数或长度为 length 的浮点数数组,每个轮廓点有一个权*/,
&beta/*曲率能量的权 Weight[s],与 alpha 类似*/,
&gamma/*图像能量的权 Weight[s],与 alpha 类似*/,
CV_VALUE/*前面三个参数的不同使用方法*/,
/*CV_VALUE 表示每个 alpha, beta, gamma 都是指向为所有点所用的一个单独数值;
CV_ARRAY 表示每个 alpha, beta, gamma 是一个指向系数数组的指针,snake 上面各点的系数都不相同。
因此,各个系数数组必须与轮廓具有同样的大小。所有数组必须与轮廓具有同样大小*/
size, criteria, calc_gradient); // 显示曲线(蓝色)
for (int prev = 0; prev < point_count; ++prev) {
int next = (prev + 1) % point_count;
// 绘线,蓝色
cv::line(src_bak, point_arr[prev], point_arr[next], cv::Scalar(255, 0, 0), 2, 8, 0);
}
}
}
}

OpenCV学习代码记录—— Snake轮廓的更多相关文章

  1. OpenCV学习代码记录——轮廓(contour)检测

    很久之前学习过一段时间的OpenCV,当时没有做什么笔记,但是代码都还在,这里把它贴出来做个记录. 代码放在码云上,地址在这里https://gitee.com/solym/OpenCVTest/tr ...

  2. OpenCV学习代码记录——人脸检测

    很久之前学习过一段时间的OpenCV,当时没有做什么笔记,但是代码都还在,这里把它贴出来做个记录. 代码放在码云上,地址在这里https://gitee.com/solym/OpenCVTest/tr ...

  3. OpenCV学习代码记录——Hough线段检测

    很久之前学习过一段时间的OpenCV,当时没有做什么笔记,但是代码都还在,这里把它贴出来做个记录. 代码放在码云上,地址在这里https://gitee.com/solym/OpenCVTest/tr ...

  4. OpenCV学习代码记录——canny边缘检测

    很久之前学习过一段时间的OpenCV,当时没有做什么笔记,但是代码都还在,这里把它贴出来做个记录. 代码放在码云上,地址在这里https://gitee.com/solym/OpenCVTest/tr ...

  5. python 设计模式学习代码记录

    @工厂模式class Beijing: def printreslut(self): print("ok") class Shanghai: def printreslut(sel ...

  6. OpenCV 学习笔记03 边界框、最小矩形区域和最小闭圆的轮廓

    本节代码使用的opencv-python 4.0.1,numpy 1.15.4 + mkl 使用图片为 Mjolnir_Round_Car_Magnet_300x300.jpg 代码如下: impor ...

  7. opencv学习笔记(二)寻找轮廓

    opencv学习笔记(二)寻找轮廓 opencv中使用findContours函数来查找轮廓,这个函数的原型为: void findContours(InputOutputArray image, O ...

  8. 【OpenCV学习笔记】三十、轮廓特征属性及应用(七)—位置关系及轮廓匹配

    http://blog.csdn.net/abc8730866/article/details/69219992 轮廓特征属性及应用(七)—位置关系及轮廓匹配 1.计算点与轮廓的距离及位置关系——po ...

  9. OpenCV学习笔记(十一) 轮廓操作

    在图像中寻找轮廓 首先利用Canny算子检测图像的边缘,再利用Canny算子的输出作为 寻找轮廓函数 findContours 的输入.最后用函数 drawContours 画出轮廓.边界Counto ...

随机推荐

  1. Ubuntu 16.04 grub rescue 模式下修复 grub

      前几天整理了下电脑的分区,合并并删除一些分区,结果导致 grub 被破坏了,Ubuntu进不去了,启动后直接进入了 rescure 模式.后来又折腾了下,终于修复好了,现总结一下. 先说一下我的系 ...

  2. Google和Baidu的站内搜索代码

    <!-- SiteSearch Google --> <form method="get" action="http://www.google.com/ ...

  3. C# xml 常规 保护 方法总结

    一 使用xsd模式文件验证xml文件: xml文件: <?xml version="1.0" encoding="utf-8" ?> <Boo ...

  4. Downloading files from a server to client, using ASP.Net, when file size is too big for MemoryStream using Generic Handlers (ashx)

    Currently, I was trying to write an ASP.Net application that involved a user clicking a ASP.Net butt ...

  5. 检索(retrieval && search )-单目标-多目标跟踪-MTMC Tracking和 ReID

    跨摄像头多目标跟踪(Multi-Target Multi-Camera Tracking, MTMC Tracking) 跨摄像头多目标跟踪(Multi-Target Multi-Camera Tra ...

  6. 细说Request与Request.Params

    在ASP.NET编程中,有三个比较常见的来自于客户端的数据来源:QueryString, Form, Cookie .我们可以在HttpRequest中访问这三大对象,比如,可以从QueryStrin ...

  7. 探索Javascript异步编程

    异步编程带来的问题在客户端Javascript中并不明显,但随着服务器端Javascript越来越广的被使用,大量的异步IO操作使得该问题变得明显.许多不同的方法都可以解决这个问题,本文讨论了一些方法 ...

  8. Eclipse中GitLab的配置和使用入门

    一.Eclipse中配置GitLab的前提条件 1.1:安装Git客户端 去官网https://git-scm.com/downloads下载合适的版本即可,一般开发环境是windows的就下载win ...

  9. 模拟日历计算 poj1008

    Maya Calendar Time Limit: 1000MS   Memory Limit: 10000K Total Submissions: 69932   Accepted: 21524 D ...

  10. Simple XOR Encryption/Decryption in C++ (And Several Other Languages)

    For details on how to implement XOR encryption using Go, see this post. If you are looking for XOR e ...