OpenCv-C++-小案例实战-切边(二)

接上篇文章…
一般的,一个扫描的文件是不可能完完全全摆正了的。它多多少少会有些倾斜。
现在有如下图片:
在这里插入图片描述
它逆时针进行了旋转,那么想要把多余的白边去掉且摆正应该怎么做呢?
步骤如下:
1、边缘检测
2、找出轮廓
3、找出最小外接矩形,获得旋转的角度
4、根据旋转的角度进行仿射变换
5、按切边的操作进行

代码(没有用到的已注释):

#include<opencv2/opencv.hpp>
#include<iostream>
#include<math.h>

using namespace cv;
using namespace std;

Mat src, dst, gray_src;

int current_level = 50;
int max_level = 255;
void cut_image(int, void*);
void rotating_image(int, void*);

const char* output_title = "rectminArea";
int main(int argc, char**argv)
{
	src = imread("D:/test/切边测试图-旋转.png", 1);
	if (src.empty())
	{
		cout << "图片未找到" << endl;
		return -1;
	}
	
	
	//namedWindow(output_title,CV_WINDOW_AUTOSIZE);
	//createTrackbar("Value", output_title, &current_level, max_level, cut_image);
	//cut_image(0,0);  //图像切边
	rotating_image(0,0); // 将图像旋转成正的
	imshow("input image", src);

	waitKey(0);
	return 0;

}
//将图像旋转成正的
void rotating_image(int, void*)
{
  /*1、边缘检测
    2、找出轮廓
    3、找出最小外接矩形,获得旋转的角度
    4、仿射变换
	5、按切边的操作进行
  */
	Mat canny_out;
	Canny(src, canny_out, current_level, 2 * current_level, 3, false);
	imshow("canny_out",canny_out);
	float minw = 0;
	float minh = 0;
	double angle = 0;

	vector<vector<Point>> contours;
	vector<Vec4i> hierarchy;
	findContours(canny_out, contours,hierarchy, RETR_TREE,CHAIN_APPROX_SIMPLE, Point(0, 0));
	Mat showImg= Mat::zeros(src.size(), CV_8UC3);;
	RNG rng(12345);
	for (size_t t = 0; t < contours.size(); t++)
	{
		RotatedRect minrect = minAreaRect(contours[t]);
		angle = minrect.angle;
		if (angle != 0)   
		{
			minh = max(minh,minrect.size.height);
	        minw = max(minw,minrect.size.width);
			
		}
	}
	printf("minw:%f\n", minw);
	printf("minh:%f\n", minh);
	printf("angle:%f\n", angle);//值为负数,说明逆时针旋转,值为正数,说明顺时针旋转
	for (size_t t = 0; t < contours.size(); t++)
	{
		RotatedRect minrect = minAreaRect(contours[t]);
		
		if (minrect.size.height== minh && minrect.size.width == minw)
		{
			
			Point2f pts[4];
			minrect.points(pts);

			for (int i = 0; i < 4; i++)
			{
				Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255));
				line(showImg, pts[i], pts[(i + 1) % 4], color, 2, 8, 0);
			}
		}
		

	}
	imshow("showImg", showImg);

	Mat dst;
	Point2f center(src.cols/2, src.rows/2);
	Mat rota = getRotationMatrix2D(center, angle, 1.0);
	warpAffine(src, dst, rota, src.size(), INTER_LINEAR,0,Scalar(255,255,255));
	imshow("correct image",dst);
	//图像切边

}



//图像切边
void cut_image(int, void*)
{
	Mat canyImg;
	cvtColor(src, gray_src, CV_BGR2GRAY);
	Canny(gray_src, canyImg, current_level, 2 * current_level, 3,false);

	vector<vector<Point>> contours;
	vector<Vec4i>hierachy;
	Mat showImg = Mat::zeros(src.size(), CV_8UC3);
	findContours(canyImg, contours, hierachy, RETR_TREE, CHAIN_APPROX_SIMPLE, Point(0, 0));
	
	RNG rng(12345);

	int minw = src.cols*0.75;
	int minh = src.rows*0.75;

	Rect box; //获取roi区域
	Mat contoursimg= Mat::zeros(src.size(), CV_8UC3);
	for (size_t t = 0; t < contours.size(); t++)
	{
		Scalar color = Scalar(rng.uniform(0, 255), rng.uniform(0, 255), rng.uniform(0, 255));
		
		drawContours(contoursimg, contours, int(t), color, 1, 8, hierachy, 0, Point(0, 0));
		
		RotatedRect minrect = minAreaRect(contours[t]); //获取轮廓的最小外接矩形
		float angle = abs(minrect.angle);
		if (minrect.size.height > minh && minrect.size.width > minw && minrect.size.width < (src.cols - 5))
		{
			Point2f pts[4];
			minrect.points(pts); //获取最小外接矩形的四个顶点坐标
			for(int i=0;i<4;i++)
			{
				line(showImg,pts[i], pts[(i + 1) % 4], color, 2, 8, 0);
				cout << "X坐标:" << minrect.center.x <<" "<< "Y坐标:" << minrect.center.y << " "<<"偏移角度:" << angle << endl;
			}
			box = minrect.boundingRect();
		}
		
		if (box.width > 0 && box.height > 0)
		{
			Mat roiImg=src(box);//截取roi区域
			imshow("roiImg", roiImg);
		}
		
	}
	
	imshow(output_title, showImg);
	imshow("contours image", contoursimg);
	

}


/* //做旋转图片的代码
int main() {
	Mat src = imread("D:/test/切边测试图.png");
	imshow("src", src);
	double angle = 45;
	Point2f center(src.cols / 2, src.rows / 2);
	Mat rot = getRotationMatrix2D(center, angle, 1);
	Rect bbox = RotatedRect(center, src.size(), angle).boundingRect();

	rot.at<double>(0, 2) += bbox.width / 2.0 - center.x;
	rot.at<double>(1, 2) += bbox.height / 2.0 - center.y;

	Mat dst;
	warpAffine(src, dst, rot, bbox.size(),1,0,Scalar(255,255,255));
	imshow("dst", dst);
	imwrite("D:/test/切边测试图-旋转.png",dst);
	waitKey(0);
	return 0;

}
*/

运行结果:
在这里插入图片描述

在这里插入图片描述

在这里插入图片描述
由此可以知道,该图片旋转的角度是-45度,也就是逆时针旋转了45度。
那么就可以进行仿射变换:

Mat dst;
	Point2f center(src.cols/2, src.rows/2);
	Mat rota = getRotationMatrix2D(center, angle, 1.0);
	warpAffine(src, dst, rota, src.size(), INTER_LINEAR,0,Scalar(255,255,255));
	imshow("correct image",dst);

最终的结果是:
在这里插入图片描述
接下来就是切边的操作了,这里参考上一篇文章:
https://blog.csdn.net/Daker_Huang/article/details/85033368

  • 2
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
课程目的:OpenCV是应用非常广泛的开源视觉处理库,在图像处理、计算机视觉和自动驾驶中有着非常重要的作用。课程设计特色:(课程当前为第一期)1、C++与Python双语教学Python语言是在计算机视觉中应用最多的一种语言,在工作中,深度学习模型的训练基本上都是使用Python语言编写的训练代码。OpenCV在这个过程中用于图像的预处理(例如图像读取、数据增强)和后处理,还可以用于显示处理的结果,功能强大,使用方便。但是在功能的部署的时候,不管是部署在服务端还是PC端,开发语言基本上用的是C++,所以如何有效的使用OpenCV进行模型或者功能的部署尤为重要。C++语言应用的好坏,在面试中可以看出一个面试者的工程实践能力的强弱,两种语言的开发掌握好了可以使工作如虎添翼。2、全模块讲解我出版了一本图书《学习OpenCV4:基于Python的算法实战》,虽然这本书是写的基于Python的算法实战,但是实际上这本书有详细的介绍算法的C++接口,还有一些C++方向的案例,是以Python为主。图书出版的时候就想双语写作,只是限于篇幅没有成行。本课程不仅采用双语教学,更是对C++的每个模块都做讲解,我们知道,很多的书其实只讲imgproc,如果你翻开一本书图像的形态学运算和图像滤波都是作为独立章节讲解的,那么这本书基本上就可以确定是只是讲解了imgproc模块,但是其他的模块在工作中也有很重要的作用。例如:core模块定义了C++的基本数据结构和基本运算(如四则运算);highgui模块是可视化与交互的模块;feature2d是特征点与特征匹配相关算法所在的模块;ml是机器学习相关的模块;dnn是深度学习相关的模块,可以使用OpenCV进行深度学习模型的部署。这些是很多的书和课程都不会讲的。3、讲解细致本课程会从环境搭建开始讲解,环境搭建尤为重要。从我多年的授课经验总结来看,如果只是给了代码,很多的入门用户环境问题处理不好的话,后面的学习很难进行下去,甚至会丧失学习的信心。4、会讲解C++和Python的开发语法问题是入门用户的一大难关,特别是C++语言。大学只是教授了C语言相关的内容,C++很多同学只懂一点皮毛,所以写代码步履维艰,我们在讲解代码的过程中会顺带讲解C++和Python的内容。我们还会讲解编译相关知识,还有库的装载与链接,这些是学校里不会教的,目前也几乎没有课程讲解。5、讲师经验丰富我讲解过C++OpenCV的多个课程,广受学员好评。我出版过两本图书《深度学习计算机视觉实战》和《学习OpenCV4》,两本书都是细致入微的讲解,主要针对的就是初学者,所以能够很好的处理课程的难易程度。6、讲义准备充分讲义准备的充分细致,标识清楚明确,重点和疑难点突出。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值