旅行好きなソフトエンジニアの備忘録

プログラミングや技術関連のメモを始めました

【Python】 ndarrayのインデキシング

書籍「科学技術計算のためのPYTHON入門」のメモです。ndarrayのインデキシングについて今まで何も意識せず使っていたのですが、書籍にまとめられていたのでメモします。

  • 基本インデキシング(ビューが生成される)
# ndarray_viewの変更がndarrayにも影響を及ぼす
ndarray_view = ndarray[1, :]
  • 応用インデキシング(コピーが生成される)
    • ブール値インデキシング
    • 整数配列インデキシング
# ブール値インデキシングの例
# 0.5より大きい箇所のみ取り出す
# maskedを変更してもndarrayに影響を及ぼさない
mask = ndarray > 0.5
masked = ndarray[mask]

# 整数配列インデキシングの例
# (0,2), (2,0), (1,3)の位置の要素を取り出す
# extractedを変更してもndarrayに影響を及ぼさない
extracted = ndarray[[0, 2, 1], [2, 0, 3]]

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

【OpenCV】 照明ムラがある環境での二値化

照明ムラがある環境下で二値化を行い対象物を抽出しようとする場合、普通に二値化処理を行うと大抵上手く抽出できません。
下の画像はその例で、画像上部が明るめ、画像下部が暗めのになっています。この画像から米粒を抜き出そうとして大津の方法を適用すると、照明ムラの影響を受けて上手く抽出できていないことが分かります。
f:id:ni4muraano:20170520213509p:plain:w200    f:id:ni4muraano:20170520213534j:plain:w200
このような時の対処法の例がMathWorksのホームページに掲載されており、今回はOpenCVで実装します。

jp.mathworks.com

// 画像をグレースケールで読み込む
Mat gray;
imread("non_uniform_illumination_example.png", IMREAD_GRAYSCALE).copyTo(gray);
if (gray.empty())
{
    throw runtime_error("Failed to open image");
}

// オープニングを実行して背景を取り出す
Mat element = getStructuringElement(MORPH_ELLIPSE, cv::Size(15, 15));
Mat background;
morphologyEx(gray, background, CV_MOP_OPEN, element);

// 前景を取得する
Mat foreground;
absdiff(gray, background, foreground);

// 二値化して米粒を抽出する
Mat binary;
threshold(foreground, binary, 0, 255, CV_THRESH_BINARY | CV_THRESH_OTSU);

f:id:ni4muraano:20170520213509p:plain
元画像

f:id:ni4muraano:20170520214658j:plain
背景

f:id:ni4muraano:20170520214708j:plain
前景

f:id:ni4muraano:20170520214715j:plain
米粒

【料理】 鶏肉のカリカリ焼き・ネギ塩レモンソース

たまに料理を作るのですが、また作る事があるかもしれないのでメモしておきます。

recipe.rakuten.co.jp

材料(2人分)

もも肉 2枚 ⇒ 1枚(350~400g、小食なので二枚は多い)
ネギ   1本
ごま油  大さじ2
塩    小さじ1/2
こしょう 適量
レモン汁 大さじ1 ⇒ 大さじ0.7位(酸っぱいと感じた)

1. フライパンを温めて、皮を下にして鶏もも肉を焼く

  • アルミホイルをひいた上から水を入れた小さい鍋を使って肉を押さえ付けるとカリカリになった
  • 7~10分焼いたと思う

2. 反対側を焼く

3. ネギ塩だれを作る

  • ネギをみじん切りにする
  • ごま油、塩、レモン汁を入れる
  • フライパンに入れ、中火でネギを少し炒める

4. 鶏肉を食べやすい大きさに切る

【OpenCV】 forループを使わずに指定した色を別の色に変更する

ディープラーニングの前処理として、画像のある色を別の色に変更しようとしていました。ただ、NumPyではforループをなるべく使用しないことが推奨されているため、どうやって実現すれば良いのか悩んでいましたが、ここ(Replace a range of colors with a specific color in python - OpenCV Q&A Forum)にアドバイスがありました。

import cv2
import numpy as np

image = cv2.imread('lenna.jpg')
# 画像の黒い部分を白に置き換える
black = [0, 0, 0]
white = [255, 255, 255]
image[np.where((image == black).all(axis=2))] = white

【Python】 行列積の演算子

行列積はNumPyのdotメソッドを利用しますが、Python3.5/NumPy1.10からは@演算子でも行列積を表現できると知ったのでメモします。

import numpy as np

A = np.array([[1, 2], [3, 4]])
B = np.array([[5, 6], [7, 8]])
# a_dot_bとa_at_bは同じ結果になります
a_dot_b = np.dot(A, B)
a_at_b = A@B

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

【Python】 csv数値データファイルの読み書き

csv数値データファイルの読み書きはnumpyを使うと簡単にできます。

import numpy as np

# csvファイルの読み込み(ヘッダがある場合はskiprowsを1にする)
data = np.loadtxt('file.csv', delimiter=',', skiprows=0)

# csvファイルへの書き込み(%0.2fで小数点以下の桁数を指定している)
np.savetxt('file.csv', data, fmt='%0.2f')

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

科学技術計算のためのPython入門 ――開発基礎、必須ライブラリ、高速化

【OpenCV】 バイラテラルフィルタの引数について

輪郭をぼかさずにノイズを除去する方法としてバイラテラルフィルタを紹介されたのですが、今一つパラメータの意味が理解できていませんでした。

void bilateralFilter(const Mat& src,
                     Mat& dst,
                     int d,
                     double sigmaColor, // 何これ?
                     double sigmaSpace, // 何これ?
                     int borderType=BORDER_DEFAULT)

幸いこのフィルタについて分かりやすく説明してくれているサイトがあったのでメモしておきます。 imagingsolution.net

サイトと対比させると、sigmaColorは\sigma_2に相当するパラメータで、大きくし過ぎるとガウシアンフィルタと何も変わらなくなるため注意が必要なパラメータ、sigmaSpaceは\sigma_1に相当するパラメータになると思います。