【Python/OpenCV】Prewittフィルタで輪郭検出

Python版OpenCVでPrewittフィルタを実装し、輪郭検出する方法をソースコード付きで解説します。

Prewittフィルタで輪郭検出

Prewitt Filter(プレヴィットフィルタ)は、画像の輪郭検出用フィルタの1つです。このフィルタは、一次微分フィルタに対して平滑化処理を加え、ノイズに対して強くなるように改良されたものです。原理と計算式については下記事で紹介しています。

【画像処理】プレヴィットフィルタの原理・特徴・計算式
画像処理におけるPrewitt(プレヴィット)フィルタの原理や特徴、計算式についてまとめました。

今回は、Prewittフィルタの処理を以下の2通りの方法で実装します。

方法①・・・cv2.filter2Dで実装
方法②・・・NumPyでアルゴリズムを書いて実装(原理の理解を深めるため)

サンプルコード① cv2.filter2Dで実装

cv2.filter2Dで実装した場合のサンプルプログラムのソースコードです。


実行結果

サンプルプログラムの実行結果です。

■入力画像(左)と出力画像(右)

kernel_xは水平方向の輪郭(輝度変化)を検出するためのカーネルです。

kernel_x = np.array([[-1, 0, 1],
                     [-1, 0, 1],
                     [-1, 0, 1]])

kernel_yは垂直方向の輪郭(輝度変化)を検出するためのカーネルです。

kernel_y = np.array([[-1, -1, -1],
                     [0, 0, 0],
                     [1, 1, 1]])

cv.filter2Dを使用して、入力画像grayに対して一次微分フィルタを適用します。gray_xは水平方向の輪郭を検出した画像です。gray_yは垂直方向の輪郭を強調した画像です。

   gray_x = cv.filter2D(gray, cv.CV_64F, kernel_x)
   gray_y = cv.filter2D(gray, cv.CV_64F, kernel_y)

水平方向と垂直方向の輪郭画像を合成して、最終的な輪郭画像dstを求めます。輪郭画像の各画素値は、水平方向と垂直方向の勾配の二乗和の平方根から求めます。

   dst = np.sqrt(gray_x ** 2 + gray_y ** 2)

サンプルコード② NumPyで実装

NumPyで実装した場合のサンプルプログラムのソースコードです。


実行結果

サンプルプログラムの実行結果です。

■入力画像(左)と出力画像(右)

コード解説

filter2d関数は、入力画像srcに対してカーネルkernelを使用して空間フィルタリングを行います。詳細は以下ページで解説しています。

【Python/OpenCV】空間フィルタリング・畳み込み演算(cv2.filter2d)
PythonとOpenCVを用いて空間フィルタリング処理する方法をソースコード付きで解説します。

注意点

先程の出力画像は正の勾配(黒→白)のみ白色になっています。
負の勾配(白→黒)も白色に塗って出力する方法は下記ページで紹介しています。

【Python/OpenCV】微分フィルタの注意点(負の値の処理)
Python版OpenCVの微分フィルタ(Sobelなど)を利用する時の注意点についてソースコード付きで解説します。

関連ページ

Python版OpenCV全般については以下ページで解説しています。

【PythonとOpenCVで画像処理超入門】使い方とサンプルコードを解説
Python版OpenCVで画像処理プログラミングを行う方法を入門者向けにソースコード付きで解説するページです。
この記事を書いた人
西住技研

Python使用歴10年以上。研究、仕事、趣味でデータ分析や作業自動化などに活用してきたノウハウを情報発信しています。
詳しいプロフィールやお問合せはこちらのページまで。
YoutubeX(旧Twitter)でも情報発信中です!

西住技研をフォローする
OpenCV

コメント