概要
OpenCV でネガポジ反転やポスタリゼーションなどの階調変換を画像に適用する方法を紹介します。
階調変換
画素値の入力を $x \in [0, 255]$ としたとき、$f(x) = y \in [0, 255]$ という関数で変換することを階調変換 (gray level transformation)、階調変換を規定する関数を階調変換関数 (gray level transformation function) といいます。
実装上は、画素値は $[0, 255]$ の 256 種類なので、入力値の変換後の値を格納した長さが 256 の 1 次元配列である LUT (Look Up Table) を用意して階調変換を行います。例えば、LUT[0] = 100
だった場合、入力値 0 は値 100 に変換されることを意味します。
cv2.LUT
OpenCV では、cv2.LUT() で階調変換が行えます。
dst = cv2.LUT(src, lut[, dst])
名前 | 型 | デフォルト値 |
---|---|---|
src | ndarray | |
入力画像 | ||
lut | sequence of int | |
Look Up Table の略。[0, 255] の値の変換後の値を (256,) の1次元配列で表す。例えば、lut[10] は入力10の変換後の値を表す。 |
名前 | 説明 | ||
---|---|---|---|
dst | 変換後の画像 |
import cv2
import numpy as np
from IPython.display import Image, display
def imshow(img):
"""ndarray 配列をインラインで Notebook 上に表示する。"""
ret, encoded = cv2.imencode(".jpg", img)
display(Image(encoded))
ネガポジ反転
画素値を反転させる階調変換をネガポジ反転といいます。
$f(x) = 255 – x$
data:image/s3,"s3://crabby-images/07f5b/07f5b4f4de053056b455d1ac2c74caf5da787104" alt=""
def invert(img):
# ネガポジ反転
y = 255 - np.arange(256)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample.jpg", cv2.IMREAD_GRAYSCALE)
dst = invert(img)
imshow(dst)
data:image/s3,"s3://crabby-images/b90a3/b90a3a22d7ff4e706ed62dfa3aaa071234ab97f6" alt=""
data:image/s3,"s3://crabby-images/f084b/f084bed95754b1480a06df4f1755f3f9b0339b19" alt=""
2 値化
閾値などの条件で入力値を 2 段階にする階調変換を2 値化 (binarization) といいます。
$$ f(x) = \begin{cases} 0 & x \le \mathrm{threshold} \\ 255 & その他の場合 \end{cases} $$data:image/s3,"s3://crabby-images/bfcc8/bfcc84cf0d42418495efcf176c10618fdc2eea42" alt=""
def binarize(img, threshold):
# 2値化
x = np.arange(256)
y = np.where(x <= threshold, 0, 255)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample.jpg", cv2.IMREAD_GRAYSCALE)
dst = binarize(img, threshold=100)
imshow(dst)
data:image/s3,"s3://crabby-images/eb9dc/eb9dcbf6877853fb0f1d94f0afbde337c8b91564" alt=""
data:image/s3,"s3://crabby-images/b3327/b332794aea38e2a66bff267ccb28bad3decdb403" alt=""
ポスタリゼーション
色を $n$ 段階にする階調変換をポスタリゼーション (postarization) といいます。
$[0, 255]$ の範囲に $n$ 個の区間を用意して、各値がどの区間に属するかを numpy.digitize()
で調べて、LUT を作成します。
data:image/s3,"s3://crabby-images/4743d/4743d0decb44c2d561116299cea37546ad50b37e" alt=""
def postalize(img, n):
# ポスタリゼーション
x = np.arange(256)
bins = np.linspace(0, 255, n + 1)
y = np.array([bins[i - 1] for i in np.digitize(x, bins)]).astype(int)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample.jpg", cv2.IMREAD_GRAYSCALE)
dst = postalize(img, n=7)
imshow(dst)
data:image/s3,"s3://crabby-images/912a0/912a0e8bddce844192b788a85ef954c0a294891a" alt=""
data:image/s3,"s3://crabby-images/bb68b/bb68bc84035586157551035cdacd7dd3d4e259ff" alt=""
ソラリゼーション
過度の露光で画像の一部が反転する現象を ソラリゼーション (solarization) といいます。 以下の関数でソラリゼーションを擬似的に再現できます。
$$ f(x) = \begin{cases} x & x \le \mathrm{threshold} \\ 255 – x & その他の場合 \end{cases} $$data:image/s3,"s3://crabby-images/733e3/733e38961d96b1f632ef1e366050c856bf31a6b9" alt=""
def solarize(img, threshold):
# ソラリゼーション
x = np.arange(256)
y = np.where(x <= threshold, x, 255 - x)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample.jpg", cv2.IMREAD_GRAYSCALE)
dst = solarize(img, threshold=128)
imshow(dst)
data:image/s3,"s3://crabby-images/23c82/23c8293cd1083612aa87868588d568e829c47a5e" alt=""
data:image/s3,"s3://crabby-images/a0986/a0986e953ab34c23a8aeaff598a9dd57aa09d972" alt=""
折れ線型トーンカーブ
折れ線で表される階調変換関数を折れ線型トーンカーブといいます。
コントラストを上げる (1)
$$ f(x) = \begin{cases} \frac{255}{t} & x \le t \\ 255 & x > t \end{cases} = \text{clip}\left(\frac{255}{t}, 0, 255\right) $$data:image/s3,"s3://crabby-images/ffd28/ffd289b2be110797832555ab0b81dac31669d110" alt=""
def increase_contrast(img, t):
x = np.arange(256)
y = np.clip(255 / t * x, 0, 255)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = increase_contrast(img, t=155)
imshow(dst)
data:image/s3,"s3://crabby-images/485b9/485b90222dce03d7165f92f28eadb10577ba9c4b" alt=""
data:image/s3,"s3://crabby-images/e4b52/e4b52957f882f1acd6513fd347c7dd2766f29f6f" alt=""
コントラストを上げる (2)
$$ f(x) = \frac{255 – t}{255} x + t $$data:image/s3,"s3://crabby-images/781df/781df0301eca37e18dd8cdd036af7c2b85177536" alt=""
def increase_contrast(img, t):
x = np.arange(256)
y = (255 - t) / 255 * x + t
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = increase_contrast(img, t=55)
imshow(dst)
data:image/s3,"s3://crabby-images/c1fb3/c1fb31a785dfab4ac04b7b9d52d74530764757fc" alt=""
data:image/s3,"s3://crabby-images/b55f7/b55f7f1f94386a10235ed94d6992717c45d9826c" alt=""
コントラストを下げる (1)
$$ f(x) = \begin{cases} 0 & x \le t \\ \frac{255 + b}{255}x – b & x > t \end{cases} = \text{clip}\left(\frac{255 + b}{255}x – b, 0, 255\right) $$ただし、$b$ は次を解いて得られる値です。
$$ \frac{255 + b}{255}t – b = 0 $$data:image/s3,"s3://crabby-images/43309/43309a75f03c3c20a7995133628a54ca27b430a6" alt=""
def decrease_contrast(img, t):
x = np.arange(256)
y = np.clip(255 / (255 - t) * x - 255 / (255 - t) * t, 0, 255)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = decrease_contrast(img, t=55)
imshow(dst)
data:image/s3,"s3://crabby-images/da5ed/da5ed23dd2117e61264acaee4c56a8c5d61807ac" alt=""
data:image/s3,"s3://crabby-images/1c2c8/1c2c81c38dbd0750e15a5989e937e4bb0a3ed9b4" alt=""
コントラストを下げる (2)
$$ f(x) = \frac{t}{255} $$data:image/s3,"s3://crabby-images/a47f9/a47f967128565c5e18700e31319a0d528871e8ed" alt=""
def decrease_contrast(img, t):
x = np.arange(256)
y = t / 255 * x
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = decrease_contrast(img, t=200)
imshow(dst)
data:image/s3,"s3://crabby-images/47330/473309ec7a7631ab9f8a8fb8bff61da41cf84e3c" alt=""
data:image/s3,"s3://crabby-images/77fff/77fffbd4e64ad06c29e41dbfd0dcbbdcf2edc705" alt=""
明るさ、コントラストの補正
$$ f(x) = \text{clip}(\alpha x + \beta , 0, 255) $$$\alpha$ はゲイン (gain) またはコントラスト (contrast)、$\beta$ はバイアス (bias) または明るさ (brightness) といいます。
data:image/s3,"s3://crabby-images/3b383/3b3838157975a603a21fee3411d770c33eb968cd" alt=""
def adjust(img, alpha, beta):
x = np.arange(256)
y = np.clip(alpha * x + beta, 0, 255)
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = adjust(img, alpha=2, beta=30)
imshow(dst)
data:image/s3,"s3://crabby-images/6e242/6e2429fba17c79808aa883a520d6364f6cdf7d45" alt=""
data:image/s3,"s3://crabby-images/34e2c/34e2c48c38104798e6d8614eae1cf95009aef5a4" alt=""
ガンマ補正
次の関数で行う階調変換をガンマ変換 (gamma transformation) またはガンマ補正 (gamma correction) といいます。
$$ f(x) = \left(\frac{x}{255}\right)^\gamma \times 255 $$$\gamma < 1$ で暗く、$\gamma > 1$ で明るくなるように補正できます。
data:image/s3,"s3://crabby-images/a8351/a835106ef68c406054d298850376d382032391e3" alt=""
def gamma_correction(img, gamma):
x = np.arange(256)
y = (x / 255) ** gamma * 255
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = gamma_correction(img, gamma=0.3)
imshow(dst)
data:image/s3,"s3://crabby-images/d8a71/d8a714cc308e970903798403f5d1826b02c4d883" alt=""
gamma=0.3
の場合
data:image/s3,"s3://crabby-images/73489/73489b91bb26a05318f6fd59c3d7f2f2cc11ec15" alt=""
gamma=3
の場合
data:image/s3,"s3://crabby-images/93e67/93e67f8825c1aa74a8ac3fe7b5ce861fe2c319ce" alt=""
S 字型トーンカーブ
S 字型のトーンカーブで階調変換すると、暗い部分は更に暗く、明るい部分は更に明るくすることができます。
以下の例では、$\arctan(x), x \in [-5, 5]$ の値を $[0, 255]$ にスケールすることで S 字型のトーンカーブを作成しました。
data:image/s3,"s3://crabby-images/8dde7/8dde7a2c97628fb49fa51032f3ac9711b90bc957" alt=""
def s_curve(img):
y = np.arctan(np.linspace(-5, 5, 256))
y = 255 / (y.max() - y.min()) * (y - y.max()) + 255
dst = cv2.LUT(img, y).astype(np.uint8)
return dst
img = cv2.imread("sample2.jpg", cv2.IMREAD_GRAYSCALE)
dst = s_curve(img)
imshow(dst)
data:image/s3,"s3://crabby-images/940b6/940b68b39785897e2b46a0a548d6906044b0d965" alt=""
data:image/s3,"s3://crabby-images/79abe/79abe327662ed1260c142d69d683e1092285f12f" alt=""
コメント