从图像中裁剪部分(使用findContours?){opencv,java}

5
我有一张带有字母方框的图片,就像这样:
Original
我已经能够裁剪出每个方框了,就像这样:
Picture of a cropped box containing a letter
现在我的问题是如何从每个方框中只裁剪出字母?所需结果看起来像这样:
desired result
我想使用findContours函数,但不确定如何做到这一点,因为它还会检测到噪音和周围的所有内容。

1
这是一个快速预览。 我看到在你所有的情况中,所需的图像都在中心.. 有变化吗?如果没有,你可以从边界上删除图像的四分之一部分并保留中心部分。如果你的情况更加多样化,请对一般情况进行一些解释,以正确理解问题。 - Pervez Alam
没有变化,字母始终位于框中心,但框将出现在不同位置。 - user3524239
1
作弊Ruzzle? :-) - Rob Audenaerde
哈哈,是的,正在制作一个乐高机器人来玩Ruzzle :) - user3524239
1个回答

14

方法 根据您可以提取框的事实,我建议采用以下方法。 如果您拥有该框,请按照以下步骤操作:

  1. 找到图像的中心
  2. 在图像中查找轮廓 - 这些可能是候选项
  3. 找到每个轮廓的边界矩形
  4. 找到每个边界矩形的中心
  5. 找到每个边界矩形与图像中心的距离
  6. 找到最小距离-答案

注意:有一个名为pad的变量,它控制结果图的填充!对于所有框都要这样做。希望这能帮到你!

祝你好运 :)


Python 代码

# reading image in grayscale
image = cv2.imread('testing2.jpg',cv2.CV_LOAD_IMAGE_GRAYSCALE)
# thresholding to get a binary one
ret, image = cv2.threshold(image, 100,255,cv2.THRESH_BINARY_INV)
# finding the center of image
image_center = (image.shape[0]/2, image.shape[1]/2)

if image is None:
    print 'can not read the image data'
# finding image contours
contours, hier = cv2.findContours(image, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
# finding distance of each contour from the center of image
d_min = 1000
for contour in contours:
    # finding bounding rect
    rect = cv2.boundingRect(contour)
    # skipping the outliers
    if rect[3] > image.shape[1]/2 and rect[2] > image.shape[0]/2:
        continue
    pt1 = (rect[0], rect[1])
    # finding the center of bounding rect-digit
    c = (rect[0]+rect[2]*1/2, rect[1]+rect[3]*1/2)
    d = np.sqrt((c[0] - image_center[0])**2 + (c[1]-image_center[1])**2)
    # finding the minimum distance from the center
    if d < d_min:
        d_min = d
        rect_min = [pt1, (rect[2],rect[3])]
# fetching the image with desired padding
pad = 5
result = image[rect_min[0][1]-pad:rect_min[0][1]+rect_min[1][1]+pad, rect_min[0][0]-pad:rect_min[0][0]+rect_min[1][0]+pad]

plt.imshow(result*255, 'gray')
plt.show()

Java 代码

    System.loadLibrary(Core.NATIVE_LIBRARY_NAME);
    // reading image 
    Mat image = Highgui.imread(".\\testing2.jpg", Highgui.CV_LOAD_IMAGE_GRAYSCALE);
    // clone the image 
    Mat original = image.clone();
    // thresholding the image to make a binary image
    Imgproc.threshold(image, image, 100, 128, Imgproc.THRESH_BINARY_INV);
    // find the center of the image
    double[] centers = {(double)image.width()/2, (double)image.height()/2};
    Point image_center = new Point(centers);

    // finding the contours
    ArrayList<MatOfPoint> contours = new ArrayList<MatOfPoint>();
    Mat hierarchy = new Mat();
    Imgproc.findContours(image, contours, hierarchy, Imgproc.RETR_EXTERNAL, Imgproc.CHAIN_APPROX_SIMPLE);

    // finding best bounding rectangle for a contour whose distance is closer to the image center that other ones
    double d_min = Double.MAX_VALUE;
    Rect rect_min = new Rect();
    for (MatOfPoint contour : contours) {
        Rect rec = Imgproc.boundingRect(contour);
        // find the best candidates
        if (rec.height > image.height()/2 & rec.width > image.width()/2)            
            continue;
        Point pt1 = new Point((double)rec.x, (double)rec.y);
        Point center = new Point(rec.x+(double)(rec.width)/2, rec.y + (double)(rec.height)/2);
        double d = Math.sqrt(Math.pow((double)(pt1.x-image_center.x),2) + Math.pow((double)(pt1.y -image_center.y), 2));            
        if (d < d_min)
        {
            d_min = d;
            rect_min = rec;
        }                   
    }
    // slicing the image for result region
    int pad = 5;        
    rect_min.x = rect_min.x - pad;
    rect_min.y = rect_min.y - pad;

    rect_min.width = rect_min.width + 2*pad;
    rect_min.height = rect_min.height + 2*pad;

    Mat result = original.submat(rect_min);     
    Highgui.imwrite("result.png", result);

编辑: 添加了Java代码!

结果

输入图像描述



看起来非常不错,你能提供Java代码吗?那太好了,谢谢!因为现在Python和Java之间存在一些差异,而且有些函数我真的不知道如何在Java中实现。 - user3524239
我几乎使用了OpenCV函数。你能告诉我哪些行对你来说很奇怪吗?! - Hadi
1
我会先尝试用Java编写,如果有进一步的问题,我会再回来的 :) - user3524239
1
干得好伙计!非常棒的帖子。 - user3524239

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接