Как найти высоту и ширину для каждого отдельного контура на изображении, используя OpenCV - PullRequest
3 голосов
/ 10 февраля 2020

img link

На изображении выше, если указана вся ширина, скажем, 30'5 ". Как рассчитать высоту и ширину для каждого отдельного контура на этом изображении, используя opencv

Ответы [ 3 ]

2 голосов
/ 10 февраля 2020

Мой подход использует minAreaRect :

#include "opencv2/highgui/highgui.hpp"
#include "opencv2/imgproc/imgproc.hpp"
#include <iostream>
#include <stdio.h>
#include <stdlib.h>

using namespace cv;
using namespace std;


int main()
{

    Mat src; Mat src_gray;
    int thresh = 100;
    RNG rng(12345);
    /// Load source image and convert it to gray
    src = imread( "/ur/img/directory/image.jpg", 1 );
    Mat original = src.clone();
    /// Convert image to gray and blur it
    cvtColor( src, src_gray, CV_BGR2GRAY );

    Mat threshold_output;
    vector<vector<Point> > contours;
    vector<Vec4i> hierarchy;

    /// Detect edges using Threshold
    threshold( src_gray, threshold_output, thresh, 255, THRESH_BINARY );
    /// Find contours
    findContours( threshold_output, contours, hierarchy, CV_RETR_TREE, CV_CHAIN_APPROX_SIMPLE, Point(0, 0) );

    /// Find the rotated rectangles for each contour
    vector<RotatedRect> minRect( contours.size() );

    for( int i = 0; i < contours.size(); i++ )
        minRect[i] = minAreaRect( Mat(contours[i]) );

    /// Draw contours + rotated rects
    Mat drawing = Mat::zeros( threshold_output.size(), CV_8UC3 );
    Mat result_zero = Mat::zeros( threshold_output.size(), CV_8UC3 );

    for( int i = 0; i< contours.size(); i++ )
    {
        Scalar color = Scalar( rng.uniform(0, 255), rng.uniform(0,255), rng.uniform(0,255) );
        // detect contours
        drawContours( drawing, contours, i, color, 1, 8, vector<Vec4i>(), 0, Point() );
        // detect rectangle for each contour
        Point2f rect_points[4]; minRect[i].points( rect_points );

        double length_1 = cv::norm(cv::Mat(rect_points[0]),cv::Mat(rect_points[1]));
        double length_2 = cv::norm(cv::Mat(rect_points[1]),cv::Mat(rect_points[2]));


        for( int j = 0; j < 4; j++ )
        {
            int temp1 = (int)length_1;
            int temp2 = (int)length_2;

            if(length_1>length_2)
                putText(original,to_string(temp1),rect_points[0],FONT_HERSHEY_SIMPLEX,1.0,Scalar(0,255,255),2);
            else
                putText(original,to_string(temp2),rect_points[0],FONT_HERSHEY_SIMPLEX,1.0,Scalar(0,255,255),2);

            line( result_zero, rect_points[j], rect_points[(j+1)%4], color, 1, 8 );
        }

    }

    /// Show in windows
    imshow("First",original);
    imshow( "Contours", drawing );
    waitKey(0);
    return(0);
}

Исходное изображение:

enter image description here

Обнаруженные прямоугольники для каждой строки:

enter image description here

Длина строки в пикселях:

enter image description here

1 голос
/ 11 февраля 2020

Чтобы получить высоту и ширину контура, вы можете использовать cv2.boundingRect. Функция возвращает информацию о контуре в виде x,y,w,h. Высота для указанного c контура будет h, а ширина будет w. Вот результат с w в пикселях, нарисованных на изображении.

enter image description here

import cv2

# Load image, convert to grayscale, Otsu's threshold
image = cv2.imread('1.jpg')
gray = cv2.cvtColor(image,cv2.COLOR_BGR2GRAY)
thresh = cv2.threshold(gray, 0, 255, cv2.THRESH_BINARY + cv2.THRESH_OTSU)[1]

# Find contours, obtain bounding rect, and draw width
cnts = cv2.findContours(thresh, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE)
cnts = cnts[0] if len(cnts) == 2 else cnts[1]
for c in cnts:
    x,y,w,h = cv2.boundingRect(c)
    cv2.putText(image, str(w), (x,y - 10), cv2.FONT_HERSHEY_SIMPLEX, 0.9, (36,255,12), 2)
    cv2.rectangle(image, (x, y), (x + w, y + h), (36,255,12), 1)

cv2.imshow('image', image)
cv2.waitKey()
0 голосов
/ 10 февраля 2020
std::vector<std::vector<cv::Point2i>> vecContours;
cv::Mat mat = cv::imread("[path to image]", cv::IMREAD_GRAYSCALE);
cv::threshold(mat, mat, 200, 255, cv::THRESH_BINARY);
cv::findContours(mat, vecContours, cv::RetrievalModes::RETR_EXTERNAL, cv::CHAIN_APPROX_NONE);
float inchPerPixel = 30.5f / mat.cols;
for (const std::vector<cv::Point2i>& vecContour : vecContours) {
    cv::Rect2i contourRect = cv::boundingRect(vecContour);
    printf("Contour width pixels : %d, width inches %f\n", contourRect.width, inchPerPixel*contourRect.width);
}

Этого можно добиться с помощью:

  1. Создание двоичного изображения с использованием порогового метода
  2. Использование метода findContours для поиска контура прямоугольников на изображении
  3. Получите размер контура прямоугольника с помощью метода boundingRect
  4. Умножьте значение контура на вычисленный коэффициент дюймов на пиксель
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...