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

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

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

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

Python

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

Q&A

1回答

2050閲覧

前景領域抽出をより正確にしたい

退会済みユーザー

退会済みユーザー

総合スコア0

OpenCV

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

Python

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

1グッド

0クリップ

投稿2019/06/26 19:48

前提・実現したいこと

下の飛行機の画像の飛行機の部分だけを取得したい
plane_256x256.jpg

発生している問題・エラーメッセージ

出力結果は,上の画像から 普通の前景領域抽出による結果, 輪郭検出による結果, 輪郭検出で作成した画像をマスクとして使用した結果 です. 一番目の画像と三番目の画像の結果も変わらず,どちらも画像の右側の空が残ってしまっている.

image1.jpg

image_canny

image2.jpg

該当のソースコード

python

1#エッジ検出画像を新たなマスクとして使う 2#ライブラリをロード 3import cv2 4import numpy as np 5from matplotlib import pyplot as plt 6 7 8#画像をロード 9image_bgr = cv2.imread('images/plane_256x256.jpg') 10image_rgb = cv2.cvtColor(image_bgr, cv2.COLOR_BGR2RGB) 11 12#矩形:始点x, 始点y, 幅, 高さ 13rectangle = (0, 56, 256, 150) 14 15#マスクの初期値を作成 16mask = np.zeros(image_rgb.shape[:2],np.uint8) 17 18#grabCutで用いる一時配列 19bgdModel = np.zeros((1, 65), np.float64) 20fgdModel = np.zeros((1, 65), np.float64) 21 22#grabCutを実行 23cv2.grabCut(image_rgb, #入力画像 24 mask, #マスク 25 rectangle, #範囲指定の矩形 26 bgdModel, #背景のための一時配列 27 fgdModel, #前景のための一時配列 28 5, #繰り返し回数 29 cv2.GC_INIT_WITH_RECT) #矩形を用いて初期化 30 31#マスクを作成.背景があることが確実もしくは高確率な場所を0に,それ以外を1に 32mask_2 = np.where((mask==2) | (mask==0), 0, 1).astype('uint8') 33 34#画像とマスクを掛け合わせて背景を除去 35image_rgb_nobg = image_rgb * mask_2[:, :, np.newaxis] 36 37#画像を表示 38plt.imshow(image_rgb_nobg), plt.axis("off") 39plt.show() 40 41image_rgb_nobg = cv2.cvtColor(image_rgb_nobg, cv2.COLOR_BGR2RGB) 42cv2.imwrite("images/image1.jpg", image_rgb_nobg) 43 44#モノクロで画像を読み込み 45image_gray = cv2.imread("images/plane_256x256.jpg", cv2.IMREAD_GRAYSCALE) 46 47#輝度の中央値を計算 48median_intensity = np.median(image_gray) 49 50#中央値0.67倍と1.33倍を閾値として設定 51lower_threshold = int(max(0, (1.0 - 0.33) * median_intensity)) 52upper_threshold = int(max(0, (1.0 + 0.33) * median_intensity)) 53 54#cannyエッジ検出を適用 55image_canny = cv2.Canny(image_gray, lower_threshold, upper_threshold) 56 57#画像を表示 58plt.imshow(image_canny, cmap="gray"), plt.axis("off") 59plt.show() 60 61 62cv2.imwrite("images/image_canny.jpg", image_canny) 63 64newmask = cv2.imread('image_canny', 0) 65 66mask[newmask == 0] = 0 67mask[newmask == 255] = 1 68 69#エッジ検出した画像でgrabCutを実行 70cv2.grabCut(image_rgb, #入力画像 71 mask, #マスク 72 None, #範囲指定の矩形 73 bgdModel, #背景のための一時配列 74 fgdModel, #前景のための一時配列 75 5, #繰り返し回数 76 cv2.GC_INIT_WITH_MASK) #矩形を用いて初期化 77 78#マスクを作成.背景があることが確実もしくは高確率な場所を0に,それ以外を1に 79mask_3 = np.where((mask==2) | (mask==0), 0, 1).astype('uint8') 80 81#画像とマスクを掛け合わせて背景を除去 82image_rgb_nobg_canny = image_rgb* mask_3[:, :, np.newaxis] 83 84#画像を表示 85plt.imshow(image_rgb_nobg_canny), plt.axis("off") 86plt.show() 87 88image_rgb_nobg_canny = cv2.cvtColor(image_rgb_nobg_canny, cv2.COLOR_BGR2RGB) 89cv2.imwrite("images/image2.jpg", image_rgb_nobg_canny)

試したこと

grabcutによる前景領域の抽出では,飛行機の右側の方の空が取れなかったため,輪郭検出で作成した画像を新たなマスクとして使用し,もう一度grabcutを行なったが結果は変わらなかった.

補足情報(FW/ツールのバージョンなど)

mac jupiter notebook python opencv

lime00👍を押しています

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

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

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

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

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

guest

回答1

0

私はPython初心者で恐縮ですが、さっき皆さんの助言を受けて作った下記のコードでこのような画像が出来ました。結構昔、人間の目で色の違いを数式で表現するための実験がされていて、その色差を利用した形になっております。輝度は反射とかあるので使っていません。下記コードの1500というパラメータは調整可能です。このような考え方もあるということで、他の考え方と組み合わせて解決で出来ないものでしょうか。

import numpy as np import cv2 import scipy.stats as sstats import matplotlib.pyplot as plt def func2(luv): l, u, v = cv2.split(luv) mode_l = sstats.mode(l[l.nonzero()])[0][0] mode_u = sstats.mode(u[l.nonzero()])[0][0] mode_v = sstats.mode(v[l.nonzero()])[0][0] mode = np.array([mode_l, mode_u, mode_v], dtype=int) a = (mode - luv)**2 b = 0.*a[:, :, 0] + a[:, :, 1] + a[:, :, 2] c = (b >= 1500).astype(np.uint8) cs = np.stack([c] * 3, axis=2) luv = luv * cs return luv bgr = cv2.imread('hik.jpg') luv = cv2.cvtColor(bgr, cv2.COLOR_BGR2Luv) luvk = func2(luv) result_k = cv2.cvtColor(luvk, cv2.COLOR_Luv2BGR) cv2.imwrite("result_k.jpg",result_k) rgb = result_k[:, :, [2, 1, 0]] plt.imshow(rgb) plt.show()

イメージ説明

投稿2021/01/17 06:30

grandchild

総合スコア10

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

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

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

まだベストアンサーが選ばれていません

会員登録して回答してみよう

アカウントをお持ちの方は

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

ただいまの回答率
85.48%

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

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

質問する

関連した質問