DevelopmentTool/OpenCV

[OpenCV] Mat 클래스 ROI 추출 함수

유제필 2022. 11. 17. 08:44

ROI(Region Of Interset)란

ROI란 이미지(영상) 에서 내가 원하는 관심 영역을 말한다.

Mat 클래스로 정의된 행렬에서 특정 사각형 영역의 부분 행렬을 추출하고 싶을 땐

괄호 연산자 재정의를 사용한다.

Mat::Mat();

Mat Mat::operator()(const Rect& roi) const;
Mat Mat::operator()(Range rowRange, Range colRange) const;

roi : 사각형 관심 영역
rowRange : 관심 행 범위
colRange : 관심 열 범위
반환값 : 추출한 부분 행렬 또는 영상이다. 부분 영사의 픽셀 데이터를 서로 공유한다.

 

부분 영상 추출 코드 작성

Mat img1 = imread("Lenna.bmp");                 // img1에 "Lenna.bmp" 파일 저장 
Mat img2 = img1(Rect(220, 120, 340, 240));      // img1 영상 특정 부분 추출

Mat 타입의 변수 img1에 레나 영상을 3채널 컬러 형태로 불러와 저장한다.

 

2번 째 행에서 img1 변수 이름 바로 뒤에 괄호를 붙여서 사용하는데,

이 부분이 Mat 클래스의 괄호 연산자 재정의로 동작한다.

 

즉, img1(Rect(220, 120, 340, 240)); 코드는 img1 영상(Lenna.bmp)의 (220, 120) 좌표부터

340 x 240 크기만큼의 사각형 부분 영상을 추출하는 코드이다.

그리고 추출한 코드를 img2 변수에 저장한다.

부분 영상을 추출할 때 주의할 점은 Mat 클래스의 괄호 연산자를 이용하여 얻은 부분 영상이

독립된 메모리 공간을 확보하여 복사하는 깊은 복사가 아니다.

픽셀 데이터를 공유하는 얖은 복사 형식이다.

 

즉, 부분 영상을 추출한 후 부분 영상의 픽셀 값을 변경하면 추출한 부분 영상뿐만

아니라 원본 영상의 픽셀 값도 함께 바뀐다.

부분 영상 추출 시 픽셀 데이터를 공유한다는 특성을 이용하면

입력 영상의 일부분에만 특정한 영상 처리를 수행할 수 있다.

Mat 행렬에서 행 또는 열 범위 추출 함수

Mat::rowRange();

지정한 범위의 행으로 구성된 행렬 반환

Mat Mat::rowRange(int startrow, int endrow) const;
Mat Mat::rowRange(const Range& r) const;

startrow : 추출항 행 범위 시작 번호(포함)
endrow : 추출할 행 범위 끝 번호(미포함)
r : 추출할 행 범위
반환값 : 지정한 행 범위에 해당하는 행렬

 

Mat::colRange()

지정한 범위의 열로 구성된 행렬 반환

Mat Mat::colRange(int startcol, int endcol) const;
Mat Mat::colRange(const Range& r) const;

startcol : 추출할 열 범위 시작 번호(포함)
endcol : 추출할 열 범위 끝 번호(미포함)
r : 추출할 열 범위
반환값 : 지정한 열 범위에 해당하는 행렬

 

Mat::row, col();

하나의 행 또는 열 추출

Mat Mat::row(int y) const;
Mat Mat::col(int x) const;

y : 부분 행렬로 추출할 행 번호
x : 부분 행렬로 추출할 열 번호
반환값 : 추출한 부분 행렬(얕은 복사)

 

마스크(Mask) 연산

상황에 따라서 사각형이 아닌 임의의 ROI 설정이 필요하기도 하다.

OpenCV에서는 임의의 모양을 갖는 ROI 설정을 위하여 일부 행렬 연산 함수에 대하여 마스크(mask) 연산을 지원한다.

 

보통 입력 영상과 크기가 같고 깊이가 CV_8U인 마스크 영상을 함께 인자로 전달받는다.

 

마스크 영상이 주어질 경우, 마스크 영상의 픽셀 값이 0이 아닌 좌표에 대해서만 연산이 수행되고,

마스크 영상은 사람의 눈으로도 구분이 쉽도록 픽셀 값이 0 또는 255인 흑백 영상이 사용된다.

 

Mat::setTo();

Mat& Mat::setTo(InputOutputArray value, InputOutputArray mask = noArray());

value : 행렬 원소에 설정할 값
mask : 마스크 행렬. 마스크 행렬의 원소가 0이 아닌 위치에서만 value 값이 설정된다.
       행렬 전체 원소 값을 설정하려면 noArray() 또는 Mat()을 지정한다.
반환값 : Mat 객체의 참조
 

Mat::setTo() 함수의 2번째 인자 mask에 마스크 영상을 지정할 수 있다.

기본 값으로 설정되어 있는 noArray()를 mask 인자로 지정하면 입력 행렬의 모든 원소 값을 Value 값으로 설정하고,

적절한 마스크 영상을 mask 인자로 지정하면 특정 영역에 대해서만 픽셀 값을 설정할 수 있다.

 

이때 마스크 영상은 Mat::setTo()를 호출하는 대상 행렬과 크기가 같아야 한다.

Mat::copyTo();

void Mat::copyTo(OutputArray m, InputArray mask) const;

m : 복사본이 저장된 행렬. 만약 *this가 행렬과 크기 및 타입이 다르면 메모리를 새로 할당한 후 픽셀 값을 복사한다.
mask : 마스크 행렬. 마스크 행렬 원소 값이 0이 아닌 좌표에서만 행렬 원소를 복사한다.
       mask 행렬은 this와 같은 크기이고 깊이는 CV_8U 이어야 한다.

Mat::copyTo() 함수는 mask 영상의 픽셀 값이 0아 아닌 위치에서만 *this 행렬 원소 값을 행렬 m으로 복사한다.

Mat::copyTo() 함수를 호출하는 *this 행렬과 인자로 전달된 m 행렬이 서로 크기 또는 타입이 같지 않을 경우,

함수 내부에서 m.create() 함수를 호출하여 대상 영상 m을 새롭게 생성한 후 마스크 영상을 고려하여 픽셀 값을 복사한다.

만약 *this 행렬과 m 행렬이 서로 크기와 타입이 같다면 m 행렬 원소 값을 그대로 유지한 상태에서 *this 행렬의 픽셀 값을 복사한다.