QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 1816|回复: 0
打印 上一主题 下一主题

OpenCV 使用分水岭算法进行图像分割

[复制链接]
字体大小: 正常 放大

1186

主题

4

听众

2922

积分

该用户从未签到

跳转到指定楼层
1#
发表于 2024-4-27 10:36 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
分水岭算法:模拟地理形态的图像分割4 j1 n# l; U" y
分水岭算法通过模拟自然地形来实现图像中物体的分类。在这一过程中,每个像素的灰度值被视作其高度,灰度值较高的像素形成山脊,即分水岭,而二值化阈值则相当于水平面,低于这个水平面的区域会被“淹没”。- }9 x% y3 w$ F
测地线距离:地形分析的核心! \, Q' B4 J/ {4 H
测地线距离是分水岭算法中的一个关键概念,它代表地球表面两点间的最短路径。这一概念在图论中同样适用,指的是图中两节点间的最短路径,与欧氏距离相比,测地线距离考虑的是实际路径。
7 j" _) w! Y- N- ?
分水岭算法的执行步骤
% |1 o: l- T, w0 w6 r0 d: n
梯度图像分类:根据灰度值对梯度图像中的像素进行分类,并设定测地距离阈值。起始点标记:选择灰度值最小的像素点作为起始点,这些点通常是局部最小值。水平面上升:随着阈值的增长,测量周围邻域像素到起始点的测地距离。若小于阈值,则淹没这些像素;若大于阈值,则在这些像素上建立“大坝”。大坝设置与区域分区:随着水平面的上升,建立更多的大坝,直到所有区域在分水岭线上相遇,完成图像的分区。避免过度分割的策略9 h- e) h- j* W
分水岭算法可能会因噪声或干扰导致图像过度分割,形成过多的小区域。解决这一问题的方法包括:
! r, `' K+ `3 [2 P5 f% w! I0 x, ?
高斯平滑:通过高斯平滑减少噪声,合并小分区。基于标记的分水岭算法:选择相对较高的灰度值像素作为起始点,手动标记或使用自动方法如距离变换来确定,从而合并小区域。OpenCV 实现 Watershed 算法函数原型:void watershed( InputArray image, InputOutputArray markers );1参数说明:image:输入的图像,必须是8位的单通道灰度图像。这个图像的梯度信息将被用来模拟水流向低洼地区流动的过程。4 t$ i( M7 D" I; ^& C9 j/ V
markers:输入输出参数,是一个与原图像大小相同的图像,用于存放分割标记。在函数调用前,这个图像应该被初始化,其中包含了用户定义的分割区域的标记。标记是通过正整数索引来表示的,表示用户已知的前景或背景区域。所有未知区域(即算法需要确定的区域)应该被标记为0。函数执行完成后,每个像素点的标记将被更新为“种子”组件的值,或者在区域边界处被设置为-1。
( i! {3 b2 J& A/ t. l
功能说明:watershed 函数会分析 image 的梯度信息,并使用 markers 中定义的已知区域作为分割的起点(种子点)。算法将从这些种子点开始,逐步对图像中的其他像素点进行区域归属的判定,直到所有像素点都被标记。在分割过程中,如果两个相邻的已知区域(种子点)相遇,算法会在它们之间创建一个边界,以避免这些区域合并在一起,从而实现分割。注意事项:markers 中的标记非常重要,它们直接影响分割的结果。因此,用户需要仔细考虑如何标记已知的前景和背景区域。分水岭算法可能会导致过度分割,特别是当图像中存在大量噪声时。在实际应用中,可能需要对图像进行预处理,如使用高斯模糊去除小的局部最小值,以减少过度分割的问题。
  1. #include <opencv2/imgcodecs.hpp>7 D4 C! y1 I- i, N1 R
  2. #include <opencv2/highgui.hpp>( y1 d# S* _6 U2 {: t; s: I! I
  3. #include <opencv2/imgproc.hpp>
    ; K$ `\" {2 E# Z\" ^

  4. 6 S: H/ \& f$ e, e( X* @- N
  5. void showImg(const std::string& windowName, const cv::Mat& img){
    ! z8 {5 m: W\" z6 u6 c
  6.     cv::imshow(windowName, img);7 B2 M7 S3 B, q: E  n9 u6 y! l% H
  7. }
    + R) K, S% X6 k( ^; }& Q$ C7 ~
  8. ' B: d2 ^# l# G# ^! B
  9. void getBackground(const cv::Mat& source, cv::Mat& dst) {2 o! r: {$ E8 E% ~. u
  10.     cv::dilate(source, dst, cv::Mat::ones(3, 3, CV_8U)); // 3x3 核
    ; V% H, K( U' R# I\" |4 d
  11. }
    \" Q; y/ C6 C4 [2 T/ S) U, x

  12. 2 b\" R; n9 J+ ?, k9 ~: A
  13. void getForeground(const cv::Mat& source, cv::Mat& dst) {' |' V9 }- b. Q8 e7 k) l- W9 M2 m
  14.     cv::distanceTransform(source, dst, cv::DIST_L2, 3, CV_32F);
    + A+ }: e  o; j
  15.     cv::normalize(dst, dst, 0, 1, cv::NORM_MINMAX);) ]: M9 b8 [( u% L2 ?5 {- C
  16. }6 M' M& ?5 ~' A9 u& w
  17. 8 H3 r8 `4 ]8 k+ v
  18. void findMarker(const cv::Mat& sureBg, cv::Mat& markers, std::vector<std::vector<cv::Point>>& contours) {
    3 n7 K9 M; @% }4 m, ]
  19.     cv::findContours(sureBg, contours, cv::RETR_EXTERNAL, cv::CHAIN_APPROX_SIMPLE);
    3 u% P- X# p$ d% V. k5 l\" h- F
  20.     // 绘制前景标记5 t7 n! k- I1 O
  21.     for (size_t i = 0, size = contours.size(); i < size; i++)1 [6 J9 U7 U\" \+ t6 z& M8 K
  22.         drawContours(markers, contours, static_cast<int>(i), cv::Scalar(static_cast<int>(i)+1), -1);
    - H& M, S, d+ F/ ]: x# G, P. v
  23. }
    % _& C# l: K. `$ n
  24. 2 q4 v- J( A# t2 v/ R; B
  25. void getRandomColor(std::vector<cv::Vec3b>& colors, size_t size) {
    0 }3 G7 U  E1 E2 [; u
  26.     for (int i = 0; i < size ; ++i) {
    : d! D9 a2 N8 I$ i# q+ K+ z\" K6 F; ]
  27.         int b = cv::theRNG().uniform(0, 256);* M+ c9 z. ^& b9 n7 r
  28.         int g = cv::theRNG().uniform(0, 256);
    . \5 u' ]  J1 g' ~/ V\" e  F, k
  29.         int r = cv::theRNG().uniform(0, 256);9 i/ J8 m0 {6 Q- A* ~
  30.         colors.emplace_back(cv::Vec3b((uchar)b, (uchar)g, (uchar)r));8 w, v7 F8 h& Q& \8 W6 y
  31.     }* x  h1 Z5 Z+ H) f; @
  32. }
    6 u+ S\" {. H( U! K
  33. / z; z3 b$ w' |7 j4 w( N
  34. int main(int argc, char** argv) {5 o: A8 T) [- A$ P; x5 _
  35.     if(argc < 2){
    : I$ n7 r. F5 h5 b' b9 P  \+ P
  36.         std::cerr << "Errorn";6 {8 f$ C( n\" F  o
  37.         std::cerr << "Provide Input Image:n n";
    \" ^% B, L# Y! C\" _% O3 A6 b. Z; T
  38.         return -1;
    ' h% h: V% l9 x' s- Z
  39.     }( i! P) d# t0 |+ J\" [7 M# s
  40.     cv::Mat original_img = cv::imread(argv[1]);
    2 G6 X5 b1 i9 _0 B4 |% m  W. q
  41.     if(original_img.empty()){
    / x% T) B( |4 I* Q: Q0 C
  42.         std::cerr << "Errorn";2 _6 F; y* b( A
  43.         std::cerr << "Cannot Read Imagen";
    $ X1 X7 d5 h# ]% P0 \1 Q
  44.         return -1;
    ) k  e* U+ h8 p' D) n  G* g, Y
  45.     }! A) [$ g+ Q- G# R, V' x! ?
  46.     cv::Mat shifted;
    \" U1 L8 e& p% P4 r% \
  47.     cv::pyrMeanShiftFiltering(original_img, shifted, 21, 51);
    1 s1 e5 H( ?: K* X3 b
  48.     showImg("Mean Shifted", shifted);. u! Q  E8 y3 T7 s4 w7 J* @; B
  49.     cv::Mat gray_img;1 f- t1 u, g\" |1 Q
  50.     cv::cvtColor(original_img, gray_img, cv::COLOR_BGR2GRAY);
    9 q% e- v0 s& g
  51.     showImg("GrayIMg", gray_img);
    8 {* O. T) K  S3 ~\" |5 W
  52.     cv::Mat bin_img;& p0 {1 t/ M# K3 ^) s6 \
  53.     cv::threshold(gray_img, bin_img, 0, 255, cv::THRESH_BINARY | cv::THRESH_OTSU);
      S6 h- V0 F$ J0 Z0 f7 Y6 B
  54.     showImg("thres img", bin_img);
    / [# p/ {9 F) V9 l
  55.     cv::Mat sure_bg;( Q) J+ z$ P' p% K  J. }
  56.     getBackground(bin_img, sure_bg);% ?, z3 l/ h9 O% f+ Y( a* L
  57.     showImg("Sure Background", sure_bg);
    ) M8 M4 _. x0 r
  58.     cv::Mat sure_fg;: _2 H; I# V% z; Z% m\" d
  59.     getForeground(bin_img, sure_fg);9 H8 i! R: `6 e% u, i5 w5 {1 T
  60.     showImg("Sure ForeGround", sure_fg);/ D: B2 s* @6 Y' c
  61.     cv::Mat markers = cv::Mat::zeros(sure_bg.size(), CV_32S);
    + j% J0 ~& x4 s8 K5 o! M# `7 q& T
  62.     std::vector<std::vector<cv::Point>> contours;3 ^5 z1 `2 P' q* U- T
  63.     findMarker(sure_bg, markers, contours);9 t\" F+ l6 C, O
  64.     cv::circle(markers, cv::Point(5, 5), 3, cv::Scalar(255), -1); // 在标记周围绘制圆圈+ a6 o) h; R# @, P. X
  65.     : V! F/ l$ D4 M! ^) k  g
  66.     cv::watershed(original_img, markers);
    . y* Y8 l7 `% v8 D8 C
  67.     cv::Mat mark;5 t# {7 J4 i8 b, R3 F6 Y! W0 D$ j) z0 P
  68.     markers.convertTo(mark, CV_8U);5 e3 R+ d\" ^+ Z5 p) _
  69.     cv::bitwise_not(mark, mark); // 将白色转换为黑色,黑色转换为白色
      q# [' {7 L  X( L: Y4 ]
  70.     showImg("MARKER", mark);4 y3 w) {) v4 S$ `\" t3 G
  71.     // 在图像中突出显示标记 /
    3 E3 H& ^3 _1 F+ x
  72.     std::vector<cv::Vec3b> colors;  t5 T* Z  V; |' R6 I1 R, a) v
  73.     getRandomColor(colors, contours.size()); // 创建结果图像
    8 @4 Q6 y; a/ ^3 z- g% C# ?
  74.     cv::Mat dst = cv::Mat::zeros(markers.size(), CV_8UC3);% i4 _4 Y# Y' P* J) p9 C  Y5 a
  75.     // 用随机颜色填充标记的对象' G1 y$ L2 P! u8 V* Z3 }
  76.     for (int i = 0; i < markers.rows; i++)( J& W. T. q7 l) V9 n7 r/ a* f( M
  77.     {- ]3 {* W8 ?) ?- W& ~( k6 O
  78.         for (int j = 0; j < markers.cols; j++)
    1 X! W\" t4 S/ Q& s
  79.         {; b; p# Y5 Z7 w! S/ {
  80.             int index = markers.at(i,j);  E. w$ O! {. E2 u1 u; B7 J* v
  81.             if (index > 0 && index <= static_cast<int>(contours.size()))
    & g3 l0 a# }5 f# _' G$ K1 }
  82.                 dst.at<cv::Vec3b>(i,j) = colors[index-1];\" r3 h% H) S; S  {4 Q6 Q
  83.         }
    % r# S6 B% P& c
  84.     }
      m4 j6 F1 c) r9 F  X
  85.     showImg("Final Result", dst);0 O; y. E0 `- x6 @\" T
  86.     cv::waitKey(0);
    / I- j1 S( g0 O. `* U) T, I
  87.     return 0;( T7 g6 P( T: o; @) Q
  88. }( W+ b1 R  \: Q9 l
复制代码
————————————————
& O! d  ?$ B5 _4 b
                            版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。                        原文链接:https://blog.csdn.net/matt45m/article/details/138211359
. p9 u/ e" P! H) Q  x
6 `7 R2 k9 g9 ?: c3 W8 |* L
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2026-4-15 07:08 , Processed in 0.337970 second(s), 51 queries .

回顶部