質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
OpenCV

OpenCV(オープンソースコンピュータービジョン)は、1999年にインテルが開発・公開したオープンソースのコンピュータビジョン向けのクロスプラットフォームライブラリです。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

Q&A

解決済

2回答

11948閲覧

OpenCV 複数の枠線がある画像から、一番大きい枠線を認識させたい

退会済みユーザー

退会済みユーザー

総合スコア0

OpenCV

OpenCV(オープンソースコンピュータービジョン)は、1999年にインテルが開発・公開したオープンソースのコンピュータビジョン向けのクロスプラットフォームライブラリです。

Python 3.x

Python 3はPythonプログラミング言語の最新バージョンであり、2008年12月3日にリリースされました。

Python

Pythonは、コードの読みやすさが特徴的なプログラミング言語の1つです。 強い型付け、動的型付けに対応しており、後方互換性がないバージョン2系とバージョン3系が使用されています。 商用製品の開発にも無料で使用でき、OSだけでなく仮想環境にも対応。Unicodeによる文字列操作をサポートしているため、日本語処理も標準で可能です。

0グッド

2クリップ

投稿2017/09/04 01:53

やりたいこと

この画像から、一番大きい枠線を認識させたい。小さい枠線は認識させない。

元の画像
イメージ説明

認識のイメージ
イメージ説明
※2値化しています。白色の部分が認識対象です。

困っていること

一番大きい枠線には小さい四角があり、どのように処理を行ったら良いか分からないこと。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

# coding:utf-8 import numpy as np import cv2 # ソース画像の読み込み(グレースケールで取得) src_img = cv2.imread("./src.jpg",0) # ソース画像を2値化(閾値を150に設定) _,bin_img = cv2.threshold(src_img,150,255,cv2.THRESH_BINARY) # 結果を表示する画像を生成(2値画像、0で初期化) res_img = np.zeros(bin_img.shape,np.uint8) # 輪郭抽出を実行 image, contours,hierarchy = cv2.findContours(bin_img, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_NONE) # 結果表示用画像に抽出した輪郭を書き込み、その中を白で埋める cv2.fillPoly(res_img,pts=[contour[-1]],color=(255,255,255)) # 結果画像の表示 cv2.imshow("Result",res_img) cv2.waitKey(0) cv2.destroyAllWindows() # 結果画像の保存 cv2.imwrite("./res.jpg",res_img)

以上の通りやれば、できるはずです。
外側の輪郭線(すなわち輪郭線を構成する点の集合)の抽出のみを目指しているのであれば、contoursが取得できた時点で終了してよいですが。

ポイントとしては、

  1. cv2.findContoursが出力する結果の第一要素(上のソースのimageに対応)は使わない
  2. cv2.findContoursの抽出モードをcv2.RETR_EXTERNALに、抽出メソッドにcv2.CHAIN_APPROX_NONEを使用すること

になる思います。

1に関しては、OpenCVのドキュメントにはimageに何が入っているのかはっきりと書かれていなかったので明言はできませんが、
中身をみるかぎり今回の目的上関係のないもの(私の環境では、bin_imgと中身が一致した)だと考えられるため、使用しませんでした。

2に関して、抽出モードをcv2.RETR_EXTERNALとするのはもっとも外側の輪郭線のみ抽出するためです。
また抽出メソッドをcv2.CHAIN_APPROX_NONEとするのは、抽出した輪郭線をすべて取得するためです。
cv2.CHAIN_APPROX_NONEとしてしまうと、間引かれた輪郭点の集合しか得られなくなってしまいます。

追記

すでに解決していたようですね。
返信欄を見逃していました。

投稿2017/09/04 12:32

編集2017/09/04 12:35
退会済みユーザー

退会済みユーザー

総合スコア0

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

退会済みユーザー

退会済みユーザー

2017/09/04 12:47

`cv2.fillPoly(res_img,contour,(255,255,255))`としたところ出来ました。 https://imgur.com/a/FjnnZ ご回答ありがとうございます。
guest

0

ベストアンサー

findContours()を使えば、解決できると思います。
http://docs.opencv.org/2.4/modules/imgproc/doc/structural_analysis_and_shape_descriptors.html?highlight=findcontours#findcontours
この場合、第2引数(mode)には、CV_RETR_EXTERNALを使うことになると思います。

輪郭の取得後、その範囲の塗りつぶしを行うには、drawContours()を使うとよいと思います。
http://docs.opencv.org/2.4/modules/imgproc/doc/structural_analysis_and_shape_descriptors.html?highlight=findcontours#drawcontours

Pythonではない上に、OpenCVのバージョンも古いですが、以下のサイトが参考になると思います。
http://imagingsolution.blog107.fc2.com/blog-entry-202.html

Pythonは詳しくないので、見当違いだったらすいません。

投稿2017/09/04 02:22

fiveHundred

総合スコア9797

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

退会済みユーザー

退会済みユーザー

2017/09/04 06:33

``` # 画像の読み込み img_rgb = cv2.imread("./sample.png") # グレースケール化 gry = cv2.cvtColor(img_rgb, cv2.COLOR_BGR2GRAY) image, contours, hierarchy = cv2.findContours(gry, cv2.RETR_EXTERNAL, cv2.CHAIN_APPROX_SIMPLE) cv2.imshow("Show Image", image) ``` で実行しましたがうまくいきません。出力画像が真っ黒になってしまいます。どうしたらいいでしょうか。
fiveHundred

2017/09/04 06:41

グレイスケールではなく、2値画像にしたらどうなりますか?
退会済みユーザー

退会済みユーザー

2017/09/04 10:09

返信ありがとうございます。 https://imgur.com/a/AuyDM このようにうまく認識できません。
退会済みユーザー

退会済みユーザー

2017/09/04 12:14

2値画像で出来ました。すいません。自分が間違っていたようです。 ありがとうございました。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問