#include<opencv2/opencv.hpp>
#include<iostream>
#include<cmath>
using namespace std;
using namespace cv;
const char input[] = "Input image";
const char output[] = "Output image";
void fileCutLine(int, void*);//对图片边缘切取
void rotateImage(int,void*);//对于图片进行旋转矫正
Mat src, dst,rotateSrc;
int main(void) {
src = imread("..\\cutLineTest.jpg");
rotateSrc = imread("..\\rotateTestImage.jpg");
if (src.empty()||rotateSrc.empty()) {
cout << "Loading image failed!" << endl;
return -1;
}
pyrDown(rotateSrc, rotateSrc);
namedWindow(input, WINDOW_AUTOSIZE);
namedWindow(output, WINDOW_AUTOSIZE);
int thresh = 134;
//createTrackbar("Control threshold", output, &thresh, 255, fileCutLine);
//fileCutLine(thresh, 0);
createTrackbar("Rotate fix", output, &thresh, 255, rotateImage);
rotateImage(thresh, 0);
//imshow(input, src);
imshow(input, rotateSrc);
waitKey(0);
return 0;
}
void fileCutLine(int thresh, void*) {
Mat gray;
Mat border;
Mat kernel = Mat(Size(3, 3), CV_8UC1);
cvtColor(src, gray, COLOR_BGR2GRAY);
GaussianBlur(gray, gray, Size(7, 7), 0);
//高斯模糊降噪
Canny(gray, border, thresh, 2 * thresh);
//通过开操作将黑色边缘上的水映文字去掉
morphologyEx(border, border, MORPH_OPEN, kernel);
vector<vector<Point>> contours;
vector<Vec4i> hierarchy;
findContours(border, contours,hierarchy, RETR_TREE, CHAIN_APPROX_SIMPLE,Point(0,0));
Rect mask;//最后获得的正确范围的遮罩
//Mat temp = Mat::zeros(src.size(), CV_8UC3);
//临时图像承载,用于观察边框
for (int i = 0; i < contours.size(); i++){
RotatedRect rect = minAreaRect(contours[i]);
Point2f corner[4];
//设置图片尺度条件,进行边缘矩形筛选
if ((rect.size.width > src.cols*0.75) && rect.size.height>src.rows*0.75) {
rect.points(corner);
//cout << "************************" << endl;
line(border, corner[0], corner[1], Scalar(0, 0, 255),2);
line(border, corner[1], corner[2], Scalar(0, 0, 255),2);
line(border, corner[3], corner[2], Scalar(0, 0, 255),2);
line(border, corner[3], corner[0], Scalar(0, 0, 255),2);
mask = rect.boundingRect();//返回包含旋转矩形的矩形
}
}
cout << mask.width << ' ' << mask.height << endl;
cout << src.cols << ' ' << src.rows << endl;
//imshow("temp", temp);
dst = src(mask);//比较好用
imshow(output, border);
if (!dst.empty()) {
imshow("Result", dst);
}
}
//如果图片是倾斜
void rotateImage(int thresh,void*) {
//Mat gray;
Mat border;
Mat kernel = Mat(Size(3, 3), CV_8UC1);
//cvtColor(rotateSrc, gray, COLOR_BGR2GRAY);
Canny(rotateSrc, border, thresh, 2 * thresh);
//通过开操作将黑色边缘上的水映文字去掉
//morphologyEx(border, border, MORPH_OPEN, kernel);
vector<vector<Point>> contours;
vector<Vec4i> hierarchy;
findContours(border, contours, hierarchy, RETR_EXTERNAL, CHAIN_APPROX_SIMPLE, Point(0, 0));
double angle = 0;//需要记录角度
Point2f center=Point2f(rotateSrc.rows/2,rotateSrc.cols/2);
Size fixSize;
Rect mask;
Mat temp = Mat(rotateSrc.size(),CV_8UC3);
for (int i = 0; i < contours.size(); i++){
RotatedRect rect = minAreaRect(contours[i]);
//center = rect.center;
Point2f corner[4];
rect.points(corner);
for (int j = 0; j < 4; j++) {
line(temp, corner[j], corner[(j + 1) % 4], Scalar(255,255,255), 2);
}
if (rect.angle != 0) {
angle = rect.angle;
cout << "Angle:" << angle << endl;
}
}
Mat martix = getRotationMatrix2D(center, angle, 1);//得到仿射矩阵
warpAffine(rotateSrc, dst, martix, rotateSrc.size());
imshow("result", dst);
imshow(output,border);
}

在调整阈值时加了traceBar,调的时候方便一些。  

结果如下(图是百度上随便找的):

opencv图像倾斜校正和切边的更多相关文章

  1. OpenCV图像金字塔:高斯金字塔、拉普拉斯金字塔与图片尺寸缩放

    这篇已经写得很好,真心给作者点个赞.题目都是直接转过来的,直接去看吧. Reference Link : http://blog.csdn.net/poem_qianmo/article/detail ...

  2. 【OpenCV新手教程之十三】OpenCV图像金字塔:高斯金字塔、拉普拉斯金字塔与图片尺寸缩放

    本系列文章由@浅墨_毛星云 出品,转载请注明出处. 文章链接:http://blog.csdn.net/poem_qianmo/article/details/26157633 作者:毛星云(浅墨) ...

  3. Opencv 图像叠加 添加水印

    Opencv 图像叠加 添加水印 C++: void Mat::copyTo(OutputArray m) const C++: void Mat::copyTo(OutputArray m, Inp ...

  4. opencv图像读取-imread

    前言 图像的读取和保存一定要注意imread函数的各个参数及其意义,尽量不要使用默认参数,否则就像数据格式出现错误(here)一样,很难查找错误原因的: re: 1.opencv图像的读取与保存; 完

  5. 学习 opencv---(12)OpenCV 图像金字塔:高斯金字塔,拉普拉斯金字塔与图片尺寸缩放

    在这篇文章里,我们一起学习下 图像金字塔 的一些基本概念,如何使用OpenCV函数pyrUp和pyrDown 对图像进行向上和向下采样,以及了解专门用于缩放图像尺寸的resize函数的用法.此博文一共 ...

  6. [OpenCV Qt教程] 在Qt图形界面中显示OpenCV图像的OpenGL Widget(第二部分)

    本文译自:http://www.robot-home.it/blog/en/software/tutorial-opencv-qt-opengl-widget-per-visualizzare-imm ...

  7. [OpenCV Qt教程] 在Qt图形界面中显示OpenCV图像的OpenGL Widget (第一部分)

    本文译自:http://www.robot-home.it/blog/en/software/tutorial-opencv-qt-opengl-widget-per-visualizzare-imm ...

  8. 关于OpenCV图像操作的默认参数问题

    本系列文章由 @yhl_leo 出品,转载请注明出处. 文章链接: http://blog.csdn.net/yhl_leo/article/details/51559490 在使用OpenCV以及其 ...

  9. OpenCV:OpenCV图像旋转的代码

    OpenCV图像旋转的代码 cv::transpose( bfM, bfM ) 前提:使用两个矩阵Mat型进行下标操作是不行的,耗费的时间太长了.直接使用两个指针对拷贝才是王道.不知道和OpenCV比 ...

随机推荐

  1. 【Android - 进阶】之Drawable简介

    Drawable是什么?Android给我们的解释是:“A general abstraction for 'something that can be drawn'.”,翻译过来就是:对于可以绘制的 ...

  2. num += num 与 num = num+ num

    a = 100def test(num): num += num print(num) test(a)print(a) 200100 这里 num += num 与 num = num+ num 不能 ...

  3. 自古逢秋悲寂寥,奈何今秋热成雕?Python使用Pyecharts统计全国温度Top10并绘图

    秋词-刘禹锡 自古逢秋悲寂寥, 我言秋日胜春朝. 晴空一鹤排云上, 便引诗情到碧霄. 古人谈及秋天,都是悲凉寂寥,那么-.我好想回到古代的秋天啊!明明到了秋天,为什么最近的气温比夏天还热. 之前做天气 ...

  4. 【开发者portal在线开发插件系列一】profile和基本上下行消息

    前言: 开发者portal支持在线开发profile(即设备建模).在线开发插件.模拟应用管理设备.模拟设备上报数据接收命令.支持离线开发的profile和插件的上传部署,是合作伙伴快速集成设备.对接 ...

  5. MySql CPU彪高到百分之1000的排查思路

    You need to enable JavaScript to run this app.   原文内容来自于LZ(楼主)的印象笔记,如出现排版异常或图片丢失等情况,可查看当前链接:https:// ...

  6. shell 循环读取文件及字符串转为数组

    文件/etc/hdocker_config内容如下: 30.72.63.94 30.72.63.95 30.72.63.96 30.72.63.97 /tmp/lasclocker.tar maste ...

  7. python使用mysql的一些坑

    注意:如果你用的是python3.x,直接去看第四个问题 遇到的第一个问题 正常来说直接执行pip安装,就是可以的,但是MySQL-python偏偏比较独特 pip install MySQL-pyt ...

  8. [TimLinux] Python nonlocal和global的作用

    1. 执行代码 以下实例都是通过执行以下代码,需要把以下执行代码放在后面实例代码的后面. a = outer_func()print("call a()") a() a() a() ...

  9. UVA-11107 Life Forms(求出现K次的子串,后缀数组+二分答案)

    题解: 题意: 输入n个DNA序列,你的任务是求出一个长度最大的字符串,使得它在超过一半的DNA序列中出现.如果有多解,按照字典序从小到大输入所有解. 把n个DNA序列拼在一起,中间用没有出现过的字符 ...

  10. .Net Core 3.0 以及其前版本编写自定义主机,以允许本机程式(转载)

    像所有的托管代码一样,.NET Core 应用程序也由主机执行. 主机负责启动运行时(包括 JIT 和垃圾回收器等组件)和调用托管的入口点. 托管 .NET Core 运行时是高级方案,在大多数情况下 ...