当前位置:   article > 正文

OpenCV图像内轮廓填-孔洞填充_opencv fill contour

opencv fill contour

图像内轮廓填充通常称为孔洞填充,主要用于目标提取。

  1. //test
  2. #include "stdafx.h"
  3. #include "cxcore.h"
  4. #include "cv.h"
  5. #include "highgui.h"
  6. // 内轮廓填充
  7. // 参数:
  8. // 1. pBinary: 输入二值图像,单通道,位深IPL_DEPTH_8U。
  9. // 2. dAreaThre: 面积阈值,当内轮廓面积小于等于dAreaThre时,进行填充。
  10. void FillInternalContours(IplImage *pBinary, double dAreaThre)
  11. {
  12. double dConArea;
  13. CvSeq *pContour = NULL;
  14. CvSeq *pConInner = NULL;
  15. CvMemStorage *pStorage = NULL;
  16. // 执行条件
  17. if (pBinary)
  18. {
  19. // 查找所有轮廓
  20. pStorage = cvCreateMemStorage(0);
  21. cvFindContours(pBinary, pStorage, &pContour, sizeof(CvContour), CV_RETR_CCOMP, CV_CHAIN_APPROX_SIMPLE);
  22. // 填充所有轮廓
  23. cvDrawContours(pBinary, pContour, CV_RGB(255, 255, 255), CV_RGB(255, 255, 255), 2, CV_FILLED, 8, cvPoint(0, 0));
  24. // 外轮廓循环
  25. int wai = 0;
  26. int nei = 0;
  27. for (; pContour != NULL; pContour = pContour->h_next)
  28. {
  29. wai++;
  30. // 内轮廓循环
  31. for (pConInner = pContour->v_next; pConInner != NULL; pConInner = pConInner->h_next)
  32. {
  33. nei++;
  34. // 内轮廓面积
  35. dConArea = fabs(cvContourArea(pConInner, CV_WHOLE_SEQ));
  36. printf("%f\n", dConArea);
  37. if (dConArea <= dAreaThre)
  38. {
  39. cvDrawContours(pBinary, pConInner, CV_RGB(255, 255, 255), CV_RGB(255, 255, 255), 0, CV_FILLED, 8, cvPoint(0, 0));
  40. }
  41. }
  42. }
  43. printf("wai = %d, nei = %d", wai, nei);
  44. cvReleaseMemStorage(&pStorage);
  45. pStorage = NULL;
  46. }
  47. }
  48. int Otsu(IplImage* src)
  49. {
  50. int height=src->height;
  51. int width=src->width;
  52. //histogram
  53. float histogram[256] = {0};
  54. for(int i=0; i < height; i++)
  55. {
  56. unsigned char* p=(unsigned char*)src->imageData + src->widthStep * i;
  57. for(int j = 0; j < width; j++)
  58. {
  59. histogram[*p++]++;
  60. }
  61. }
  62. //normalize histogram
  63. int size = height * width;
  64. for(int i = 0; i < 256; i++)
  65. {
  66. histogram[i] = histogram[i] / size;
  67. }
  68. //average pixel value
  69. float avgValue=0;
  70. for(int i=0; i < 256; i++)
  71. {
  72. avgValue += i * histogram[i]; //整幅图像的平均灰度
  73. }
  74. int threshold;
  75. float maxVariance=0;
  76. float w = 0, u = 0;
  77. for(int i = 0; i < 256; i++)
  78. {
  79. w += histogram[i]; //假设当前灰度i为阈值, 0~i 灰度的像素(假设像素值在此范围的像素叫做前景像素) 所占整幅图像的比例
  80. u += i * histogram[i]; // 灰度i 之前的像素(0~i)的平均灰度值: 前景像素的平均灰度值
  81. float t = avgValue * w - u;
  82. float variance = t * t / (w * (1 - w) );
  83. if(variance > maxVariance)
  84. {
  85. maxVariance = variance;
  86. threshold = i;
  87. }
  88. }
  89. return threshold;
  90. }
  91. int main()
  92. {
  93. IplImage *img = cvLoadImage("c://temp.jpg", 0);
  94. IplImage *bin = cvCreateImage(cvGetSize(img), 8, 1);
  95. int thresh = Otsu(img);
  96. cvThreshold(img, bin, thresh, 255, CV_THRESH_BINARY);
  97. FillInternalContours(bin, 200);
  98. cvNamedWindow("img");
  99. cvShowImage("img", img);
  100. cvNamedWindow("result");
  101. cvShowImage("result", bin);
  102. cvWaitKey(-1);
  103. cvReleaseImage(&img);
  104. cvReleaseImage(&bin);
  105. return 0;
  106. }

原图:



二值化之后的图 && 填充之后的图:

特意 在大月亮里面 涂了 两个大点点 !哈哈哈

是为了要看看 cvFindContours 函数中返回的 pContour 中的元素是如何存储的。

嗯, 看见了么? O(∩_∩)O哈哈~

pContour ->h_next 连接的是图中的四个大轮廓: 一个大月亮, 两颗小星星 和一个小白点。

pContour->v_next 连接的是每个大轮廓下面的 孩子们 。 pContour->v_next 指向第一个孩子,第二个孩子 在 第一个孩子 的 水平 方向 而不是 垂直方向, 所以 第二个孩子是

pContour->v_next->h_next 哦~~~

所以 大月亮轮廓里面有 2个 小轮廓, 就是我们要找的内轮廓。 找到 之后, 当其 面积 < 200个像素 的时候,才对其进行填充 。。。。 这样做是为了考虑到, 当 内部轮廓不是 目标空洞 而是 目标背景时, 防止填错 。。。。

声明:本文内容由网友自发贡献,不代表【wpsshop博客】立场,版权归原作者所有,本站不承担相应法律责任。如您发现有侵权的内容,请联系我们。转载请注明出处:https://www.wpsshop.cn/w/凡人多烦事01/article/detail/72431
推荐阅读
相关标签
  

闽ICP备14008679号