OpenCV+Python图像分割

分水岭算法

用于分割多个相邻的物体。

原理

灰度图像根据灰度值可以把像素之间的关系看成山峰和山谷的关系,高亮度(灰度值高)的地方是山峰,低亮度的地方是山谷。给每个孤立的山谷(局部最小值)不同颜色的水(label),当水涨起来,根据周围的山峰(梯度),不同的山谷也就是不同的颜色会开始合并,要避免这个,可以在水要合并的地方建立障碍,直到所有山峰都被淹没。所创建的障碍就是分割结果,这个就是分水岭的原理,但是这个方法会分割过度,因为有噪点,或者其他图像上的错误。所以OpenCV实现的分水岭算法,可以指定哪些是要合并的点,哪些不是,我们要做的是给不同的标签。给我们知道是前景或者是目标用一种颜色加上标签,给我们知道是背景或者非目标加上另一个颜色,最后不知道是什么的区域标记为0。
这篇文章翻译地很好了:
OpenCV-Python教程:31.分水岭算法对图像进行分割

markers = cv2.watershed(img, markers)
# img: 8-bit 3-channel image
# markers: 32-bit single-channel
# markers(input): 0表示未知区域, 其他正数表示确定是不同物体的标签
# markers(output): -1表示是边界
# 初始化marker的基本套路
img = cv2.imread('xxx.png')
gray = cv2.cvtColor(img, cv2.COLOR_BGR2GRAY)
ret, thresh = cv2.threshold(gray, 0, 255, cv2.THRESH_BINARY_INV+cv2.THRESH_OTSU)
# noise removal
kernel = np.ones((3,3), np.uint8)
opening = cv2.morphologyEx(thresh, cv2.MORPH_OPEN, kernel, iterations = 2)

# sure background area
sure_bg = cv2.dilate(opening, kernel, iterations=3)

# Finding sure foreground area
dist_transform = cv2.distanceTransform(opening, cv2.DIST_L2, 5)
ret, sure_fg = cv2.threshold(dist_transform, 0.7*dist_transform.max(), 255, 0)

# Finding unknown region
sure_fg = np.uint8(sure_fg)
unknown = cv2.subtract(sure_bg, sure_fg)

# Marker labelling
ret, markers = cv2.connectedComponents(sure_fg)

# Add one to all labels so that sure background is not 0, but 1
markers = markers + 1

# Now, mark the region of unknown with zero
markers[unknown==255] = 0

GrabCut算法

Grabcut算法论文
原理博文
用于分割出图像中的前景。

mask = cv2.grabCut(img, mask, rect, bgdModel, fgdModel, iterCount, mode=None)
"""
img:   8-bit 3-channel image
mask:  8-bit single-channel 最终的结果,shape和img一样
       当mode设置为cv2.GC_INIT_WITH_MASK时,需要用户自己对mask进行初始化
       矩阵里面只有四个值:
              GC_BGD    = 0,  //背景
              GC_FGD    = 1,  //前景 
              GC_PR_BGD = 2,  //可能背景
              GC_PR_FGD = 3   //可能前景 
rect: 一块矩形区域,矩形外面一定是背景,当mode设置为cv2.GC_INIT_WITH_RECT时,才会使用
bgdModel, fgdModel: 算法过程中用到的临时矩阵,不用管,传None即可
mode: 
    cv2.GC_INIT_WITH_RECT
    cv2.GC_INIT_WITH_MASK
"""

RECT可以通过findContour等方式获得
获得mask之后,从源图像中提取前景的方法

mask2 = np.where((mask == 2) | (mask == 0), 0, 1).astype('uint8')
img = img * mask2[:, :, np.newaxis]

你可能感兴趣的:(OpenCV+Python图像分割)