OpenCv實戰2 切邊處理

2 切邊處理

1.1 題目:

將下圖作圖的圖片灰色區域切邊,並將圖片旋轉爲正。結果爲右圖。

在這裏插入圖片描述

1.2 思路

1、 轉化爲灰度圖像,
2、 進行邊緣檢測,找出邊緣
3、 邊緣中查找輪廓
4、 輪廓中查找面積最大的外接矩形
5、 使用矩形角度,仿射進行旋轉
6、 再次邊緣加測,輪廓查找,找出輪廓的垂直邊界最大矩形。
7、 提取ROI區域

1.3 示例代碼

int main(int argc, char *argv[])
{
	Mat sorImg = imread(path.toStdString() + "qiebian.jpg", IMREAD_UNCHANGED);
	imshow("sor", sorImg);

	// 第一步:轉化爲灰度圖像
	Mat grayImg;
	cvtColor(sorImg, grayImg, COLOR_BGR2GRAY);

	// 第二步:進行邊緣檢測,找出邊緣
	Mat cannyImg;
	Canny(grayImg, cannyImg, 20, 40);
	imshow("cannyImg", cannyImg);

	//第三步:邊緣中查找輪廓
	vector<vector<Point>> controus;
	vector<Vec4i>ContourTree;
	findContours(cannyImg, controus, ContourTree, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0,0));

	// 第四步:輪廓中查找最大外接外接矩形,找出矩形角度
	int size = controus.size();
	RotatedRect maxRotatedRect;
	RotatedRect Rotatedrect;
	for (int i = 0; i < size; ++i)
	{
		Rotatedrect = minAreaRect(controus[i]);
		if (Rotatedrect.boundingRect().area() > maxRotatedRect.boundingRect().area())
		{
			maxRotatedRect = Rotatedrect;
		}
	}

	// 第五步:使用矩形角度,對原圖像進行旋轉
	Mat rotateImg;
	float angle = maxRotatedRect.angle;
	Mat kenerl = getRotationMatrix2D(Point2f(cannyImg.cols/2, cannyImg.rows/2),360+ angle,1);
	warpAffine(sorImg, rotateImg, kenerl, sorImg.size(),INTER_LINEAR);

	// 第六步:再次邊緣加測,輪廓查找,找出輪廓的垂直邊界最大矩形
	cvtColor(rotateImg, grayImg, COLOR_BGR2GRAY);
	Canny(grayImg, cannyImg, 20, 40);
	findContours(cannyImg, controus, ContourTree, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
	size = controus.size();
	Rect maxRect;
	Rect rect;
	for (int i = 0; i < size; ++i)
	{
		rect = boundingRect(controus[i]);
		if (rect.area() > maxRect.area())
		{
			maxRect = rect;
		}
	}

	// 第七步:提取ROI區域,得到最終結果
	Mat dstImg = rotateImg(maxRect);
	imshow("dstImg", dstImg);
	waitKey();

	return 0;	
}

1.4 關鍵API

Canny:邊緣檢測。
findContours:查找輪廓。
minAreaRect:計算輪廓的最小外接矩形。
boundingRect:計算輪廓的垂直邊界最小矩形。
warpAffine:仿射旋轉圖像。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章