ホームページ >バックエンド開発 >Python チュートリアル >Python でドキュメント スキャナーを構築するにはどうすればよいですか?

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

王林
王林転載
2023-04-26 13:10:111772ブラウズ

翻訳者 | Bugatti

レビュアー | Sun Shujuan

物理的なスペースを節約したり、バックアップを作成したりするために、ドキュメントをデジタル化したい場合があります。いずれにせよ、紙の文書の写真を準形式に変換するプログラムを作成することは、まさに Python が得意とすることです。

適切なライブラリを組み合わせて使用​​すると、ドキュメントをデジタル化するための小さなアプリケーションを構築できます。プログラムは、物理文書の画像を入力として受け取り、それにいくつかの画像処理手法を適用して、入力のスキャンされたバージョンを出力します。

1. 環境を準備する

まず、Python の基本を理解し、NumPy Python ライブラリの使用方法を知る必要があります。 。

任意の Python IDE を開き、2 つの Python ファイルを作成します。一方に main.py という名前を付け、もう一方にtransform.py という名前を付けます。次に、ターミナル上で次のコマンドを実行して、必要なライブラリをインストールします。

pip install OpenCV-Python imutils scikit-image NumPy

OpenCV-Python を使用して画像入力を取得し、画像処理を行い、Imutils を使用して入力画像と出力画像のサイズを変更し、scikit-image を使用して画像のしきい値を設定します。 NumPy は配列を扱うのに役立ちます。

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

#インストールが完了し、IDE がプロジェクトのバックボーンを更新するまで待ちます。バックボーン コンテンツが更新されたら、プログラミングを開始できます。完全なソース コードは GitHub リポジトリにあります。

2. インストールされているライブラリをインポートします

main.py ファイルを開き、インストールされているライブラリをインポートします。これにより、必要に応じてそれらの機能を呼び出して使用できるようになります。

import cv2
import imutils
from skimage.filters import threshold_local
from transform import perspective_transform

perspective_transform によってスローされたエラーを無視します。 transform.py ファイルの処理が完了すると、エラーは消えます。

3. 入力サイズの取得と調整

スキャンするドキュメントの鮮明な画像を撮影します。文書の四隅すべてとその内容が見えるようにしてください。プログラム ファイルが保存されているのと同じフォルダーにイメージをコピーします。

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

# 入力画像のパスを OpenCV に渡します。遠近法変換時に必要となるため、元の画像のコピーを作成します。元の画像の高さを、サイズを変更したい高さで割ります。これによりアスペクト比が維持されます。最後に、調整された画像が出力されます。

# Passing the image path
original_img = cv2.imread('sample.jpg')
copy = original_img.copy()

# The resized height in hundreds
ratio = original_img.shape[0] / 500.0
img_resize = imutils.resize(original_img, height=500)

# Displaying output
cv2.imshow('Resized image', img_resize)

# Waiting for the user to press any key
cv2.waitKey(0)

上記のコードの出力は次のとおりです:

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

これで、元の画像の高さが 500 ピクセルに調整されました。

4. 調整した画像をグレースケール画像に変換します

調整した RGB 画像をグレースケール画像に変換します。ほとんどの画像処理ライブラリは、処理が簡単なため、グレースケール画像のみを処理します。

gray_image = cv2.cvtColor(img_resize, cv2.COLOR_BGR2GRAY)
cv2.imshow('Grayed Image', gray_image)
cv2.waitKey(0)

元のイメージとグレースケール イメージの違いに注目してください。

#IDE 上で灰色のイメージを表示するプログラム出力Python でドキュメント スキャナーを構築するにはどうすればよいですか?

カラー テーブルが白黒のテーブルになります。

5. エッジ検出器を使用する

ガウスぼかしフィルターをグレースケール画像に適用してノイズを除去します。次に、OpenCV canny 関数を呼び出して、画像内に存在するエッジを検出します。

blurred_image = cv2.GaussianBlur(gray_image, (5, 5), 0)
edged_img = cv2.Canny(blurred_image, 75, 200)
cv2.imshow('Image edges', edged_img)
cv2.waitKey(0)

出力にエッジが表示されます。

Python でドキュメント スキャナーを構築するにはどうすればよいですか?#これから作業するエッジは、ドキュメントのエッジです。

6. 最大の輪郭を見つける

エッジ画像の輪郭を検出します。最大 5 つの等高線のみを残して降順に並べ替えます。輪郭を周期的にソートすることにより、最大の四辺輪郭が近似的に得られる。

cnts, _ = cv2.findContours(edged_img, cv2.RETR_LIST, 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.02 * peri, True)

if len(approx) == 4:
doc = approx
break

4 辺があるアウトラインには、ドキュメントが含まれる可能性があります。

7. 文書の輪郭の四隅を丸で囲みます

検出された文書の輪郭の四隅を丸で囲みます。これは、プログラムが画像内の文書を検出できるかどうかを判断するのに役立ちます。

p = []

for d in doc:
tuple_point = tuple(d[0])
cv2.circle(img_resize, tuple_point, 3, (0, 0, 255), 4)
p.append(tuple_point)

cv2.imshow('Circled corner points', img_resize)
cv2.waitKey(0)

調整された RGB 画像のいくつかの角を丸で囲みます。

#ドキュメントを検出したら、画像からドキュメントを抽出する必要があります。 Python でドキュメント スキャナーを構築するにはどうすればよいですか?

8. ワープ パースペクティブを使用して目的の画像を取得する

ワープ パースペクティブは、画像を変換して歪みを修正するために使用されるコンピューター ビジョン技術です。画像をさまざまな平面に変換し、さまざまな角度から画像を表示できるようにします。

warped_image = perspective_transform(copy, doc.reshape(4, 2) * ratio)
warped_image = cv2.cvtColor(warped_image, cv2.COLOR_BGR2GRAY)
cv2.imshow("Warped Image", imutils.resize(warped_image, height=650))
cv2.waitKey(0)

歪んだ画像を取得するには、透視変換を実行する単純なモジュールを作成する必要があります。

9. 変換モジュール

该模块将对文档角的点进行排序。它还会将文档图像转换成不同的平面,并将相机角度更改为俯拍。

打开之前创建的那个transform.py文件,导入OpenCV库和NumPy库。

import numpy as np
import cv2

这个模块将含有两个函数。创建一个对文档角点的坐标进行排序的函数。第一个坐标将是左上角的坐标,第二个将是右上角的坐标,第三个将是右下角的坐标,第四个将是左下角的坐标。

def order_points(pts):
 # initializing the list of coordinates to be ordered
 rect = np.zeros((4, 2), dtype = "float32")

 s = pts.sum(axis = 1)

 # top-left point will have the smallest sum
 rect[0] = pts[np.argmin(s)]

 # bottom-right point will have the largest sum
 rect[2] = pts[np.argmax(s)]

 '''computing the difference between the points, the
 top-right point will have the smallest difference,
 whereas the bottom-left will have the largest difference'''
 diff = np.diff(pts, axis = 1)
 rect[1] = pts[np.argmin(diff)]
 rect[3] = pts[np.argmax(diff)]

 # returns ordered coordinates
 return rect

创建将计算新图像的角坐标,并获得俯拍的第二个函数。然后,它将计算透视变换矩阵,并返回扭曲的图像。

def perspective_transform(image, pts):
 # unpack the ordered coordinates individually
 rect = order_points(pts)
 (tl, tr, br, bl) = rect

 '''compute the width of the new image, which will be the
 maximum distance between bottom-right and bottom-left
 x-coordinates or the top-right and top-left x-coordinates'''
 widthA = np.sqrt(((br[0] - bl[0]) ** 2) + ((br[1] - bl[1]) ** 2))
 widthB = np.sqrt(((tr[0] - tl[0]) ** 2) + ((tr[1] - tl[1]) ** 2))
 maxWidth = max(int(widthA), int(widthB))

 '''compute the height of the new image, which will be the
 maximum distance between the top-left and bottom-left y-coordinates'''
 heightA = np.sqrt(((tr[0] - br[0]) ** 2) + ((tr[1] - br[1]) ** 2))
 heightB = np.sqrt(((tl[0] - bl[0]) ** 2) + ((tl[1] - bl[1]) ** 2))
 maxHeight = max(int(heightA), int(heightB))

 '''construct the set of destination points to obtain an overhead shot'''
 dst = np.array([
[0, 0],
[maxWidth - 1, 0],
[maxWidth - 1, maxHeight - 1],
[0, maxHeight - 1]], dtype = "float32")

 # compute the perspective transform matrix
 transform_matrix = cv2.getPerspectiveTransform(rect, dst)

 # Apply the transform matrix
 warped = cv2.warpPerspective(image, transform_matrix, (maxWidth, maxHeight))

 # return the warped image
 return warped

现在您已创建了转换模块。perspective_transform导入方面的错误现在将消失。

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

注意,显示的图像有俯拍。

10、运用自适应阈值,保存扫描输出

在main.py文件中,对扭曲的图像运用高斯阈值。这将给扭曲的图像一个扫描后的外观。将扫描后的图像输出保存到含有程序文件的文件夹中。

T = threshold_local(warped_image, 11, offset=10, method="gaussian")
warped = (warped_image > T).astype("uint8") * 255
cv2.imwrite('./'+'scan'+'.png',warped)

以PNG格式保存扫描件可以保持文档质量。

11、显示输出

输出扫描后文档的图像:

cv2.imshow("Final Scanned image", imutils.resize(warped, height=650))
cv2.waitKey(0)
cv2.destroyAllWindows()

下图显示了程序的输出,即扫描后文档的俯拍。

Python でドキュメント スキャナーを構築するにはどうすればよいですか?

12、计算机视觉在如何进步?

创建文档扫描器涉及计算机视觉的一些核心领域,计算机视觉是一个广泛而复杂的领域。为了在计算机视觉方面取得进步,您应该从事有趣味又有挑战性的项目。

您还应该阅读如何将计算机视觉与当前前技术结合使用方面的更多信息。这让您能了解情况,并为所处理的项目提供新的想法。

原文链接:https://www.makeuseof.com/python-create-document-scanner/

以上がPython でドキュメント スキャナーを構築するにはどうすればよいですか?の詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。

声明:
この記事は51cto.comで複製されています。侵害がある場合は、admin@php.cn までご連絡ください。