18
17

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

More than 3 years have passed since last update.

ハッシュ値で画像の類似度を判定する

Posted at

概要

2つの画像のハッシュ値をそれぞれ求め、ハッシュ値間の類似度を求めます。
完全に一致しているかどうか、ではなく類似度が分かるので、しきい値を適切に設定するとあいまいな画像一致判定ができます。
本稿はC#を前提として記述していますが、使用しているアルゴリズムは他の言語でも実装があるようなので、一定の参考になると思います。

前提

前述のとおり本稿はC#を前提として記述しています。
このアルゴリズムを使うには、CoenM.ImageSharp.ImageHashをNuGetでインストールし、以下のnamespaceを追加してください。

using CoenM.ImageHash;
using CoenM.ImageHash.HashAlgorithms;
using SixLabors.ImageSharp;
using SixLabors.ImageSharp.PixelFormats;

ライブラリ

ハッシュ値を求めるアルゴリズム

CoenM.ImageSharp.ImageHashでは、pHash(Perceptual Hash)、aHash(Average Hash)、dHash(Difference Hash)の3つのアルゴリズムを使用できます。

pHashとaHashは以下の記事に詳しいです。
同一画像を判定するためのハッシュ化アルゴリズム

以下は、pHashとaHashの開発者であるDr.Neal Krawetz氏のブログのエントリから各アルゴリズムの評価を意訳したものです。

  • pHashは遅いが、正確性では最高のパフォーマンスを発揮する。
  • aHashは高速だが、正確性に欠ける。一致すべき画像の取りこぼしは少ないが、一致すべきでない画像と一致するケースが多い。
  • dHashはaHashと同等の速度で、正確性もかなり高い。

※ブログのコメントに「pHash別に遅くないじゃん。二次元のDCTをO(n^4)でやってるから遅い。O(n^3)でやればpHashも全然速いじゃん。」みたいな指摘がありますが、CoenM.ImageSharp.ImageHashがどっちなのかは分かりません…。

画像のハッシュ値を求める

この画像のハッシュ値を求めます。(©いらすとや)
neko_kaburu_man.png

コード

    // 画像の読み込み
    Image<Rgba32> image = Image.Load<Rgba32>(@"neko_kaburu_man.png");
    // ハッシュアルゴリズムのインスタンス化(この例ではpHash)
    IImageHash pHashArgorithm = new PerceptualHash();
    // ハッシュ値を求める
    ulong pHash = pHashArgorithm.Hash(image);
    // 出力
    Console.WriteLine($"pHash = {pHash}");

実行結果

pHash = 10402329587663758416

pHashがPerceptual Hashでのハッシュ値です。
CoenM.ImageSharp.ImageHashではulongですが、アルゴリズム本来の意味では64bitのハッシュが求まります。

画像間の類似度を求める

この2つの画像のハッシュ値それぞれを求め、類似度を求めます。
neko_kaburu_man.pngneko_kaburu_woman.png

コード

    // 画像の読み込み
    Image<Rgba32> image1 = Image.Load<Rgba32>(@"neko_kaburu_man.png");
    Image<Rgba32> image2 = Image.Load<Rgba32>(@"neko_kaburu_woman.png");

    // ハッシュアルゴリズムのインスタンス化(この例ではpHash)
    IImageHash pHashArgorithm = new PerceptualHash();

    // ハッシュを求める
    ulong pHash1 = pHashArgorithm.Hash(image1);
    ulong pHash2 = pHashArgorithm.Hash(image2);

    // ハッシュ値間の類似度を求める
    double pHashSimilarity = CompareHash.Similarity(pHash1 , pHash2);

    // 出力
    Console.WriteLine($"pHash1 = {pHash1}");
    Console.WriteLine($"pHash2 = {pHash2}");
    Console.WriteLine($"pHashSimilarity = {pHashSimilarity}");

実行結果

pHash1 = 10402329587663758416
pHash2 = 10401203732853990480
pHashSimilarity = 90.625

pHash1が左の画像のハッシュ値、pHash2が右の画像のハッシュ値です。
pHashSimilarityが類似度です。
類似度について詳しくは後述します。

いろんなアルゴリズムでハッシュ値を求める

コード

    // 画像の読み込み
    Image<Rgba32> image = Image.Load<Rgba32>(@"neko_kaburu_man.png");
    // ハッシュアルゴリズムのインスタンス化
    IImageHash pHashArgorithm = new PerceptualHash();
    IImageHash aHashArgorithm = new AverageHash();
    IImageHash dHashArgorithm = new DifferenceHash();
    // ハッシュ値を求める
    ulong pHash = pHashArgorithm.Hash(image);
    ulong aHash = aHashArgorithm.Hash(image);
    ulong dHash = dHashArgorithm.Hash(image);
    // 出力
    Console.WriteLine($"pHash = {pHash}");
    Console.WriteLine($"aHash = {aHash}");
    Console.WriteLine($"dHash = {dHash}");

実行結果

pHash = 10402329587663758416
aHash = 11590717229577021695
dHash = 5958916077812472516

pHash、aHash、dHashともに使い方は同じです。
ハッシュアルゴリズムをインスタンス化する際に使いたいアルゴリズムを選んでください。

アルゴリズムごとの類似度サンプル(以下コード省略)

いろんなケースでアルゴリズムごとに類似度を求めてみます。
類似度は100で完全一致です。
一致と判断するのはおおむね90台後半にしたほうがいいです。

一部だけ異なる画像の比較

neko_kaburu_man.pngneko_kaburu_woman.png

実行結果

pHashSimilarity = 90.625
aHashSimilarity = 67.1875
dHashSimilarity = 92.1875

異なる部分が小さければある程度類似画像とみなしてくれます。
が、しきい値を下げすぎることはお勧めしません。
この例で一致と判断したいのであれば、異なっていない箇所だけ切り出して比較したほうがいいです。
(異なっている箇所が分かる場合だけですが)

異なる画像の比較

neko_kaburu_man.pngpose_dance_ukareru_woman.png

実行結果

pHashSimilarity = 50
aHashSimilarity = 59.375
dHashSimilarity = 54.6875

全然違う画像と比較した場合の類似度は50前後です。
類似度0は全く逆の画像(aHashならネガポジ反転とか)なので、ある意味元の画像と近いです。

リサイズされた画像の比較

neko_kaburu_man.pngneko_kaburu_man_resized.png

実行結果

pHashSimilarity = 100
aHashSimilarity = 100
dHashSimilarity = 100

これらのアルゴリズムは、同じサイズに縮小してからハッシュ値を求めるというものなので、リサイズされた画像やアスペクト比が異なる画像との比較もできます。

ウォーターマークありなしの比較

neko_kaburu_man.pngneko_kaburu_man_watermarked.png

実行結果

pHashSimilarity = 96.875
aHashSimilarity = 62.5
dHashSimilarity = 90.625

ウォーターマークありなしもある程度類似画像とみなしてくれます。
aHashは類似度が低いですね。

圧縮ノイズで劣化した画像との比較

neko_kaburu_man_good.jpgneko_kaburu_man_bad.jpg

実行結果

pHashSimilarity = 96.875
aHashSimilarity = 100
dHashSimilarity = 100

圧縮ノイズがあっても類似画像とみなしてくれます。

一部切り出した画像との比較

neko_kaburu_man.pngneko_kaburu_man_cropped.png

実行結果

pHashSimilarity = 53.125
aHashSimilarity = 70.3125
dHashSimilarity = 59.375

一部切り出した画像は基本的に異なる画像とみなされます。
アルゴリズムによって多少異なりますが、ハッシュ値は縦横8x8といった、ごく画像にリサイズしてから計算します。
一部切り出されていると縮小した際に全く異なった結果になるため、類似度が低くなります。
余白が広くなっていたり狭くなっていたりしても同様です。

色味が補正された画像の比較

neko_kaburu_man.pngneko_kaburu_man_retoned.png

実行結果

pHashSimilarity = 96.875
aHashSimilarity = 82.8125
dHashSimilarity = 93.75

程度によりますが、色味が補正されていても類似画像とみなしてくれます。
これもaHashは類似度が低いですね。

人間の目にはよく似ているように見える画像の比較

pose_dance_ukareru_man.pngpose_dance_ukareru_woman.png

実行結果

pHashSimilarity = 78.125
aHashSimilarity = 87.5
dHashSimilarity = 75

aHashは比較的高いように見えますが、類似度90未満は信用しないほうがいいです。
こういう画像の比較は機械学習等の手段を検討すべきです。

一見同じに見えるが、背景の透過率が異なっている画像の比較

neko_kaburu_man.pngneko_kaburu_man_good.jpg

実行結果

pHashSimilarity = 43.75
aHashSimilarity = 51.5625
dHashSimilarity = 46.875

左の画像はPNGで右の画像はJPEGです。
フォーマットの違いは重要ではありません。
左の画像は背景が透明で、右の画像は白色になっています。
CoenM.ImageHash.HashAlgorithmsでは透過度も含めてハッシュ値を求めているため、異なる画像とみなされます。
透過度を無視したければ、ハッシュ値を求める前に透過情報をなくしておく必要があります。

雑感

  • サンプルのように、用途によって最適なアルゴリズムは異なります。正確性を重視したい場合はpHashがよさそうです。
  • 今回は余白の多いイラストで試しましたが、写真などの余白がない(情報量が多い)画像であればもっと顕著な結果が出ます。
  • どの類似度以上なら一致と判断するかどうかは実際に比較してチューニングが必要です。情報量が多く、ノイズが少ない画像であれば類似度98あたりをしきい値にするといいと思います。
18
17
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
18
17

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?