使用 OpenCV 进行文档矫正
使用 OpenCV 进行文档矫正
std::vector<std::vector<cv::Point>> cvhelper::findCorners(const cv::Mat &image) {
cv::Mat gaussImage;
cv::GaussianBlur(image, gaussImage, cv::Size(5, 5), 0);
cv::threshold(gaussImage, gaussImage, 0, 255, cv::THRESH_BINARY | cv::THRESH_OTSU);
cv::Canny(gaussImage, gaussImage, 50, 200);
cv::morphologyEx(gaussImage, gaussImage, cv::MORPH_CLOSE,
cv::Mat::ones(5, 5, CV_32F));
std::vector<std::vector<cv::Point>> contours;
std::vector<cv::Vec4i> hierarchy;
cv::findContours(gaussImage, contours, hierarchy, cv::RETR_LIST, cv::CHAIN_APPROX_SIMPLE);
std::vector<std::vector<cv::Point>> ret;
for (auto &contour: contours) {
double peri = cv::arcLength(contour, true);
std::vector<cv::Point> approx;
cv::approxPolyDP(contour, approx, 0.05 * peri, true);
if (approx.size() == 4 && cv::isContourConvex(approx) && cv::contourArea(approx) > 1000) {
ret.push_back(approx);
}
}
return ret;
}
void cvhelper::GetDocumentRect(cv::Mat &src, std::vector<cv::Point> &ret_points) {
int shrunkImageHeight = 500;
cv::Mat shrunkImage;
cv::resize(src, shrunkImage, cv::Size(shrunkImageHeight * src.cols / src.rows, shrunkImageHeight));
cv::cvtColor(shrunkImage, shrunkImage, cv::COLOR_BGR2Luv);
std::vector<cv::Mat> channels;
cv::split(shrunkImage, channels);
std::vector<std::vector<cv::Point>> documentCorners;
for (const auto &channel: channels) {
std::vector<std::vector<cv::Point>> corners = findCorners(channel);
if (!corners.empty()) {
double maxArea = 0.0;
std::vector<cv::Point> maxAreaContourIt;
for (const auto &contour: corners) {
double area = cv::contourArea(contour);
if (area > maxArea) {
maxArea = area;
maxAreaContourIt = contour;
}
}
if (maxArea > 0) {
std::vector<cv::Point> scaledContour;
for (const auto &point: maxAreaContourIt) {
cv::Point scaledPoint;
scaledPoint.x = static_cast<int>(point.x * static_cast<double>(src.rows) /
shrunkImageHeight);
scaledPoint.y = static_cast<int>(point.y * static_cast<double>(src.rows) /
shrunkImageHeight);
scaledContour.push_back(scaledPoint);
}
documentCorners.push_back(scaledContour);
}
}
}
if (documentCorners.size() == 0) {
return;
}
std::vector<cv::Point> maxDocumentCorners;
double maxDocumentArea = 0.0;
for (const auto &documentCorner: documentCorners) {
double area = cv::contourArea(documentCorner);
if (area > maxDocumentArea) {
maxDocumentArea = area;
maxDocumentCorners = documentCorner;
}
}
ret_points = maxDocumentCorners;
}
cv::Mat cvhelper::CutKeyPosition(cv::Mat &src, std::vector<cv::Point2f> &src_points) {
// 透视变换
cv::Point2f dst_points[4];
dst_points[0] = cv::Point2f(src.cols, 0);
dst_points[1] = cv::Point2f(0, 0);
dst_points[2] = cv::Point2f(0, src.rows);
dst_points[3] = cv::Point2f(src.cols, src.rows);
cv::Point2f tL = src_points[1];
cv::Point2f tR = src_points[0];
cv::Point2f bR = src_points[3];
cv::Point2f bL = src_points[2];
int width = (std::min)(cv::norm(tR - tL), cv::norm(bR - bL));
int height = (std::min)(cv::norm(tR - bR), cv::norm(tL - bL));
cv::Mat M = cv::getPerspectiveTransform(src_points.data(), dst_points);
cv::Mat dst;
cv::warpPerspective(src, dst, M, src.size());
// 缩放图像
cv::resize(dst, dst, cv::Size(width, height));
return dst;
}
效果展示
使用 OpenCV 进行文档矫正的更多相关文章
- 配置允许匿名用户登录访问vsftpd服务,进行文档的上传下载、文档的新建删除等操作
centos7环境下 临时关闭防火墙 #systemctl stop firewalld 临时关闭selinux #setenforce 0 安装ftp服务 #yum install vsftpd - ...
- 怎样在Linux下使用Markdown进行文档工作
怎样在Linux下使用Markdown进行文档工作 在Linux系统中,编辑markdown能够用retext工具: sudo apt-get install retext retext Releas ...
- Linux系统通过FTP进行文档基本操作【华为云分享】
[摘要] Linux系统里通过FTP可以对文档进行上传,更改权限和基本的文档管理. 获得Linux系统后,不熟悉命令操作的情况下,可以通过FTP工具进行文档操作,下面以WinSCP工具为例进行讲解: ...
- SharePoint2016如何使用策略进行文档归档
前言 最近项目用户需要提供文档按照日期或标题关键字进行对应的文档归档操作,为了实施这个操作,需要准备2个文档库,我这里准备了如下文档库: 1. 测试文档库:在测试文档中上传几篇文档,如下图: 2. 我 ...
- 使用Word进行文档修订版本的比较
项目经理在实际的工作过程中,比如要写文档方案,就需要对文档的修订版本进行管理和控制.在以前的工作中,笔者使用的是UltraEdit这个软件工具中的Ultra Compare这个子工具来进行的文档版本的 ...
- 利用kibana插件对Elasticsearch进行文档和索引的CRUD操作
#添加索引PUT lagou { "settings": { "index": { , } } }#查看 索引设置 GET lagou/_settings GE ...
- 机器学习实战之朴素贝叶斯进行文档分类(Python 代码版)
贝叶斯是搞概率论的.学术圈上有个贝叶斯学派.看起来吊吊的.关于贝叶斯是个啥网上有很多资料.想必读者基本都明了.我这里只简单概括下:贝叶斯分类其实就是基于先验概率的基础上的一种分类法,核心公式就是条件概 ...
- sharepoint2013用场管理员进行文档库的爬网提示"没有权限,拒绝"的解决方法
爬网提示被拒绝,场管理员明明可以打开那个站点的,我初步怀疑是:环回请求(LoopbackRequest)导致的 解决方法就是修改环回问题.修改注册表 具体操作方法: http://www.c-shar ...
- Marior去除边距和迭代内容矫正用于自然文档矫正
一.简要介绍 本文简要介绍了论文" Marior: Margin Removal and Iterative Content Rectification for Document Dewar ...
- 相机标定过程(opencv) + matlab参数导入opencv + matlab标定和矫正
%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% 辛苦原创所得,转载请注明出处 %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% ...
随机推荐
- python模块Configparser读取 ini(cfg,txt)等配置文件
一个ini文件的组成: 一个ini文件是由多个section组成,每个section中以key=vlaue形式存储数据: 示例 # 安装 pip install ConfigParser # 1.导包 ...
- 问题:django中对datetime类型数据在pycharm中sqlite3进行修改时,修改后datetime日期数据变成了时间戳类型
这是正在修改的 提交完之后 问题原因 问题原因是sqlite数据库对日期类型不敏感,Pycharm直接插入会变成图中这样的时间戳,用POST请求添加数据或Django自带的后台管理插入不会有这样的问题 ...
- windows下的批处理bat文件和Linux下的shell文件的含义
原文:https://www.cnblogs.com/caiguodong/p/10308255.html shell(Linux.Solaris) bat(windows) 含义 # rem 注释行 ...
- 06、etcd 写请求执行流程
本篇内容主要来源于自己学习的视频,如有侵权,请联系删除,谢谢. 上一节我们学习了 etcd 读请求执行流程,这一节,我们来学习 etcd 写请求执行流程. 1.etcd写请求概览 etcd 一个写请求 ...
- 【Azure 应用服务】NodeJS项目部署在App Service For Linux环境中,部署完成后应用无法访问
问题描述 NodeJS项目部署在App Service For Linux环境中,部署完成后应用无法访问,在启动日志中发现错误消息:ERROR - Container XXX_0 for site X ...
- 手把手教你蜂鸟e203协处理器的扩展
NICE协处理器 赛题要求: 对蜂鸟E203 RISC-V内核进行运算算子(譬如加解密算法.浮点运算.矢量运算等)的扩展,可通过NICE协处理器接口进行添加,也可直接实现RISC-V指令子集(譬如 ...
- adb monkey 有哪些参数?
adb monkey 是 Android Debug Bridge (ADB) 工具中的一个命令,用于执行随机事件来对 Android 应用进行压力测试.以下是 adb monkey 命令的一些常用参 ...
- 二十一: Mysql 锁机制
Mysql 锁机制 事务的 隔离性 由这章讲述的 锁 来实现. 1. 概述 锁是计算机协调多个进程或线程并发访问某一资源的机制.在程序开发中会存在多线程同步的问题,当多个线程并发访问某个数据的时候,尤 ...
- vscode ctrl + 鼠标左键 点击 不进入组件,弹出一个框再点才进入的问题 目录名和vue文件名 开头不能大写 需从起生效
vscode ctrl + 鼠标左键 点击 不进入组件,弹出一个框再点才进入的问题 目录名和vue文件名 开头不能大写 需从起生效 就是这个,很讨厌,原来是开头不能大写字母的问题造成的. 小写字母就没 ...
- Win10使用Dism++离线安装.Net3.5
.Net3.5的安装包在Win10已经不能使用了,在线安装.Net3.5会很卡(跟网络无关),最好是使用Dism++提取Win10系统镜像文件离线安装. 打开Dism++软件,按照如下步骤操作: 选择 ...