python实现图片方向纠正
bigegpt 2024-12-15 11:27 3 浏览
目前有两种方案:
1、图片种类(发票、表格、单据...)较少,可直接使用基于模版的方向纠正
2、图片种类较多,基于图片分类模型预测 ,然后进行图片纠正
实现:
1、基于模版的方向纠正代码如下:
根据模版的方向纠正代码
# -*- coding: utf-8 -*-
import numpy as np
import cv2
"""
模式匹配,匹配出身份证信息,并且矫正身份证信息;
单应性指的是图像在投影发生了畸变后仍然能够有较高的检测和匹配准确率;
"""
class findidcard:
def __init__(self):
pass
# 身份证正面截取,img1为身份证模板, img2为需要识别的图像
def front_find(self, img2_name):
#身份证背面识别
img1_name = './picture_templates/invoice_mask.png'
# 最小匹配数量设为10个, 大于这个数量从中筛选出10个最好的
MIN_MATCH_COUNT = 10
# 读入两幅图片 图片中有相同部分
img1 = cv2.UMat(cv2.imread(img1_name, 0)) # queryImage in Gray
img1 = self.img_resize(img1, 640)
img2 = cv2.UMat(cv2.imread(img2_name, 0)) # trainImage in Gray
img2 = self.img_resize(img2, 1920)
img_org = cv2.UMat(cv2.imread(img2_name))
img_org = self.img_resize(img_org, 1920)
# 获取sift特征检测器
sift = cv2.xfeatures2d.SIFT_create()
# 使用SIFT查找关键点和描述符
kp1, des1 = sift.detectAndCompute(img1, None)
kp2, des2 = sift.detectAndCompute(img2, None)
# kdtree建立索引方式的常量参数
FLANN_INDEX_KDTREE = 0
index_params = dict(algorithm=FLANN_INDEX_KDTREE, trees=5)
search_params = dict(checks=10) # checks指定索引树要被遍历的次数
flann = cv2.FlannBasedMatcher(index_params, search_params)
# 进行匹配搜索
matches = flann.knnMatch(des1, des2, k=2)
# 两个最佳匹配之间距离需要大于ratio 0.7,距离过于相似可能是噪声点
# 寻找距离近的放入good列表
good = []
for m, n in matches:
if m.distance < 0.7 * n.distance:
good.append(m)
# reshape为(x,y)数组
# 如果足够多,就筛选
if len(good) > MIN_MATCH_COUNT:
# 通过距离近的描述符 找到两幅图片的关键点
src_pts = np.float32([kp1[m.queryIdx].pt for m in good]).reshape(-1, 1, 2)
dst_pts = np.float32([kp2[m.trainIdx].pt for m in good]).reshape(-1, 1, 2)
# 单应性匹配图关键点匹配线。。不懂啥意思
# 用HomoGraphy计算图像与图像之间映射关系, M为转换矩阵
M, mask = cv2.findHomography(src_pts, dst_pts, cv2.RANSAC, 5.0)
# 使用转换矩阵M计算出img1在img2的对应形状
h, w = cv2.UMat.get(img1).shape
M_r = np.linalg.inv(M)
im_r = cv2.warpPerspective(img_org, M_r, (w, h))
# cv2.imwrite("middle.jpg", im_r)
else:
print("Not enough matches are found - %d/%d" % (len(good), MIN_MATCH_COUNT))
return im_r
def showimg(img):
cv2.namedWindow("contours", 0)
cv2.resizeWindow("contours", 1280, 720)
cv2.imshow("contours", img)
cv2.waitKey()
# 身份证背面截取,img1为身份证模板, img2为需要识别的图像
def back_find(self, img2_name):
# 身份证背面识别
img1_name = './picture_templates/idcard_back_mask.jpg'
# 最小匹配数量设为10个, 大于这个数量从中筛选出10个最好的
MIN_MATCH_COUNT = 10
# 读入两幅图片 图片中有相同部分
img1 = cv2.UMat(cv2.imread(img1_name, 0)) # queryImage in Gray
img1 = self.img_resize(img1, 640)
img2 = cv2.UMat(cv2.imread(img2_name, 0)) # trainImage in Gray
img2 = self.img_resize(img2, 1920)
img_org = cv2.UMat(cv2.imread(img2_name))
img_org = self.img_resize(img_org, 1920)
# 获取sift特征检测器
sift = cv2.xfeatures2d.SIFT_create()
# 使用SIFT查找关键点和描述符
kp1, des1 = sift.detectAndCompute(img1, None)
kp2, des2 = sift.detectAndCompute(img2, None)
# kdtree建立索引方式的常量参数
FLANN_INDEX_KDTREE = 0
index_params = dict(algorithm=FLANN_INDEX_KDTREE, trees=5)
search_params = dict(checks=10) # checks指定索引树要被遍历的次数
flann = cv2.FlannBasedMatcher(index_params, search_params)
# 进行匹配搜索
matches = flann.knnMatch(des1, des2, k=2)
# 两个最佳匹配之间距离需要大于ratio 0.7,距离过于相似可能是噪声点
# 寻找距离近的放入good列表
good = []
for m, n in matches:
if m.distance < 0.7 * n.distance:
good.append(m)
# reshape为(x,y)数组
# 如果足够多,就筛选
if len(good) > MIN_MATCH_COUNT:
# 通过距离近的描述符 找到两幅图片的关键点
src_pts = np.float32([kp1[m.queryIdx].pt for m in good]).reshape(-1, 1, 2)
dst_pts = np.float32([kp2[m.trainIdx].pt for m in good]).reshape(-1, 1, 2)
# 单应性匹配图关键点匹配线。。不懂啥意思
# 用HomoGraphy计算图像与图像之间映射关系, M为转换矩阵
M, mask = cv2.findHomography(src_pts, dst_pts, cv2.RANSAC, 5.0)
# 使用转换矩阵M计算出img1在img2的对应形状
h, w = cv2.UMat.get(img1).shape
M_r = np.linalg.inv(M)
im_r = cv2.warpPerspective(img_org, M_r, (w, h))
else:
print("Not enough matches are found - %d/%d" % (len(good), MIN_MATCH_COUNT))
return im_r
#重置图片大小
def img_resize(self, imggray, dwidth):
crop = imggray
size = crop.get().shape
height = size[0]
width = size[1]
height = height * dwidth / width
crop = cv2.resize(src=crop, dsize=(dwidth, int(height)), interpolation=cv2.INTER_CUBIC)
return crop
if __name__ == "__main__":
idfind = findidcard()
result = idfind.front_find('./testimages/invoice5.png')
# idfind.showimg(result)
cv2.namedWindow("contours", 0)
cv2.resizeWindow("contours", 1280, 720)
cv2.imshow("contours", result)
cv2.waitKey()
2、基于方向模型预测实现方向纠正
方向纠正模型训练使用百度飞浆paddle开源的paddleClas进行模型训练。训练样本可使用业务中的具体数据,准备0度图片样本1000张,然后使用脚本对其进行90、180、270度方向旋转,分别放到0、90、180、270 文件夹,使用脚本生成coco数据集。修改paddleClas的ppcls/configs/ImageNet/PPLCNetV2/PPLCNetV2_base_direction.yaml 配置文件,更改样本地址路径和类别,进行训练。最终使用模型进行方向预测,通过预测能得出图片的旋转方向,程序再次对图片进行旋转纠正。
预测结果:
进行图片旋转纠正:
try:
im = Image.open(img_path)
except Exception as e:
print(str(e))
else:
if im.mode == "RGBA":
im = im.convert('RGB')
os.remove(img_path)
im.save(img_path)
im = Image.open(img_path)
if rorate_radio == 90:
rimg = im.transpose(Image.ROTATE_90)
if rorate_radio == 180:
rimg = im.transpose(Image.ROTATE_180)
if rorate_radio == 270:
rimg = im.transpose(Image.ROTATE_270)
if rorate_radio == 0:
rimg = im
img = dst+"_"+img
print("target img:",img)
rimg.save(os.path.join(dst, img))
效果:
相关推荐
- 了解Linux目录,那你就了解了一半的Linux系统
-
大到公司或者社群再小到个人要利用Linux来开发产品的人实在是多如牛毛,每个人都用自己的标准来配置文件或者设置目录,那么未来的Linux则就是一团乱麻,也对管理造成许多麻烦。后来,就有所谓的FHS(F...
- Linux命令,这些操作要注意!(linux命令?)
-
刚玩Linux的人总觉得自己在演黑客电影,直到手滑输错命令把公司服务器删库,这才发现命令行根本不是随便乱用的,而是“生死簿”。今天直接上干货,告诉你哪些命令用好了封神!喜欢的一键三连,谢谢观众老爷!!...
- Linux 命令速查手册:这 30 个高频指令,拯救 90% 的运维小白!
-
在Linux系统的世界里,命令行是强大的武器。对于运维小白而言,掌握一些高频使用的Linux命令,能极大提升工作效率,轻松应对各种系统管理任务。今天,就为大家奉上精心整理的30个Linu...
- linux必学的60个命令(linux必学的20个命令)
-
以下是Linux必学的20个基础命令:1.cd:切换目录2.ls:列出文件和目录3.mkdir:创建目录4.rm:删除文件或目录5.cp:复制文件或目录6.mv:移动/重命名文件或目录7....
- 提高工作效率的--Linux常用命令,能够决解95%以上的问题
-
点击上方关注,第一时间接受干货转发,点赞,收藏,不如一次关注评论区第一条注意查看回复:Linux命令获取linux常用命令大全pdf+Linux命令行大全pdf为什么要学习Linux命令?1、因为Li...
- 15 个实用 Linux 命令(linux命令用法及举例)
-
Linux命令行是系统管理员、开发者和技术爱好者的强大工具。掌握实用命令不仅能提高效率,还能解锁Linux系统的无限潜力,本文将深入介绍15个实用Linux命令。ls-列出目录内容l...
- Linux 常用命令集合(linux常用命令全集)
-
系统信息arch显示机器的处理器架构(1)uname-m显示机器的处理器架构(2)uname-r显示正在使用的内核版本dmidecode-q显示硬件系统部件-(SMBIOS/DM...
- Linux的常用命令就是记不住,怎么办?
-
1.帮助命令1.1help命令#语法格式:命令--help#作用:查看某个命令的帮助信息#示例:#ls--help查看ls命令的帮助信息#netst...
- Linux常用文件操作命令(linux常用文件操作命令有哪些)
-
ls命令在Linux维护工作中,经常使用ls这个命令,这是最基本的命令,来写几条常用的ls命令。先来查看一下使用的ls版本#ls--versionls(GNUcoreutils)8.4...
- Linux 常用命令(linux常用命令)
-
日志排查类操作命令查看日志cat/var/log/messages、tail-fxxx.log搜索关键词grep"error"xxx.log多条件过滤`grep-E...
- 简单粗暴收藏版:Linux常用命令大汇总
-
号主:老杨丨11年资深网络工程师,更多网工提升干货,请关注公众号:网络工程师俱乐部下午好,我的网工朋友在Linux系统中,命令行界面(CLI)是管理员和开发人员最常用的工具之一。通过命令行,用户可...
- 「Linux」linux常用基本命令(linux常用基本命令和用法)
-
Linux中许多常用命令是必须掌握的,这里将我学linux入门时学的一些常用的基本命令分享给大家一下,希望可以帮助你们。总结送免费学习资料(包含视频、技术学习路线图谱、文档等)1、显示日期的指令:d...
- Linux的常用命令就是记不住,怎么办?于是推出了这套教程
-
1.帮助命令1.1help命令#语法格式:命令--help#作用:查看某个命令的帮助信息#示例:#ls--help查看ls命令的帮助信息#netst...
- Linux的30个常用命令汇总,运维大神必掌握技能!
-
以下是Linux系统中最常用的30个命令,精简版覆盖日常操作核心需求,适合快速掌握:一、文件/目录操作1.`ls`-列出目录内容`ls-l`(详细信息)|`ls-a`(显示隐藏文件)...
- Linux/Unix 系统中非常常用的命令
-
Linux/Unix系统中非常常用的命令,它们是进行文件操作、文本处理、权限管理等任务的基础。下面是对这些命令的简要说明:**文件操作类:*****`ls`(list):**列出目录内容,显...
- 一周热门
- 最近发表
- 标签列表
-
- mybatiscollection (79)
- mqtt服务器 (88)
- keyerror (78)
- c#map (65)
- resize函数 (64)
- xftp6 (83)
- bt搜索 (75)
- c#var (76)
- mybatis大于等于 (64)
- xcode-select (66)
- mysql授权 (74)
- 下载测试 (70)
- linuxlink (65)
- pythonwget (67)
- androidinclude (65)
- logstashinput (65)
- hadoop端口 (65)
- vue阻止冒泡 (67)
- oracle时间戳转换日期 (64)
- jquery跨域 (68)
- php写入文件 (73)
- kafkatools (66)
- mysql导出数据库 (66)
- jquery鼠标移入移出 (71)
- 取小数点后两位的函数 (73)