我有 1000 张旧明信片要扫描,我认为使用某种自动裁剪/旋转工具优化我的工作流程可能是个好主意,因此我开始使用 Python 研究 openCV。
可以想象,我的目标是从这张图片中创建 3 张图片,每张图片包含一张明信片。我已经尝试了许多 opencv 选项,到目前为止我能得到的最好的代码是:
import cv2, sys, imutils
cv2.namedWindow('image', cv2.WINDOW_NORMAL)
image = cv2.imread("sample1600.jpg")
ratio = image.shape[0] / 300.0
image = imutils.resize(image, height = 800)
gray = cv2.cvtColor(image, cv2.COLOR_BGR2GRAY)
gray = cv2.GaussianBlur(gray, (5, 5), 0)
ret, th = cv2.threshold(gray,220,235,1)
edged = cv2.Canny(th, 25, 200)
(cnts, _) = cv2.findContours(edged.copy(), cv2.RETR_TREE, cv2.CHAIN_APPROX_SIMPLE)
cnts = sorted(cnts, key = cv2.contourArea, reverse = True)[:5]
for c in cnts:
peri = cv2.arcLength(c, True)
approx = cv2.approxPolyDP(c, 0.05 * peri, True)
if len(approx) == 4:
cv2.drawContours(image, [approx], -1, (0, 255, 0), 3)
cv2.imshow("Image", image)
cv2.waitKey(0)
这段代码的问题是:
- 它没有找到离边框太近的底部图像;
- 它仅适用于我的测试图像,但似乎不是很通用。例如,“ret, th = cv2.threshold(gray,220,235,1)”行将阻止事情处理我认为具有不同直方图的图像。
有没有人知道使此代码更好地工作并更通用以满足我处理扫描图像的要求的最佳方法?
编辑:我最初没有提到并且可能有用的是单个明信片的宽度和高度之间的比率应该近似为 √2。情况并非总是如此,但如果我的脚本能够有效地处理这种类型的明信片,我会非常高兴(它们代表了我收藏的 > 99%)
编辑 24/04:感谢@Riccardo,我现在有了一个适用于我的第一个示例图像的脚本,因此添加一个新的脚本以尝试找到更强大的解决方案:
编辑 24/04 #2:由于@Riccardo 为前两个样本提供了非常有效的解决方案,因此由于第一个样本的图像之间的空间有限,这里还有另外两个似乎有点复杂: