我有一个如下所示的二进制图像。如何删除 Blob 边框上的单个像素?
如果您不想这样做,则无需提供完整的代码,您可以解释一些算法或为我指明正确的方向。
最佳答案
我设法使用midtiby建议的点击和错过转换解决了这个问题。我使用以下内核来检测顶部,右侧,底部和左侧的单个像素。
-1 -1 -1 1 -1 -1 1 1 1 -1 -1 1
-1 1 -1 1 1 -1 -1 1 -1 -1 1 1
1 1 1 1 -1 -1 -1 -1 -1 -1 -1 1
其中
-1
表示背景,1
表示前景,而0
表示无关(在这种情况下不使用)。四个命中和未命中变换的结果将用作删除单个像素的 mask 。以下是Python/OpenCV中的完整代码:
import numpy as np
import cv2
def hitmiss(src, kernel):
im = src / 255
k1 = (kernel == 1).astype('uint8')
k2 = (kernel == -1).astype('uint8')
e1 = cv2.erode(im, k1, borderType=cv2.BORDER_CONSTANT)
e2 = cv2.erode(1-im, k2, borderType=cv2.BORDER_CONSTANT)
return e1 & e2
if __name__ == "__main__":
im = cv2.imread('blobs.png', cv2.CV_LOAD_IMAGE_GRAYSCALE)
_, im_binary = cv2.threshold(im, 50, 255, cv2.THRESH_BINARY)
kernel = np.array([[-1,-1, 1],
[-1, 1, 1],
[-1,-1, 1]])
im_mask = np.zeros(im_binary.shape, np.uint8)
im_mask |= hitmiss(im_binary, kernel)
im_mask |= hitmiss(im_binary, np.fliplr(kernel))
im_mask |= hitmiss(im_binary, kernel.T)
im_mask |= hitmiss(im_binary, np.flipud(kernel.T))
im_dst = im_binary & ((1 - im_mask) * 255)
cv2.imwrite('dst.png', im_dst)
给定此输入图像:
该脚本将产生以下输出: