Android 图片相似度比较-SIFT&直方图

因为项目需要,最近研究了一下在Android App内实现两张图片的相似度比较。查了一圈之后发现OpenCV有提供SIFT算法,所以最终决定用SIFT算法实现两张图片的相似度比较。

在正式开始之前,对与OpenCV中提供的SIFT算法先简单说明一下,以免大家像我一样走很多弯路。因为SIFT算法版权的原因,在3.0版本开始,SIFT算法被移出了官方提供的SDK。需要用的人要自己下载Opencv_contrib源码包进行编译。这个过程可以在网上查到,比较繁琐。但比较巧的是这个SIFT算法的版权刚好最近到期,所以官方最近又把SIFT算法加进了SDK中。所以想用的朋友可以直接下载OpenCV 4.4.0版本就可以了。具体OpenCV的接入方法可以参考我的另外一篇文章(https://www.jianshu.com/p/6fdefcdb86de),在这就不再赘述了。

接下来,我们就直接进入正题,如何用SIFT算法实现两张图片的相似度比较。话不多说,直接上代码,看了就自然会了。

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.util.Log;
import com.wildma.idcardcamera.R;
import org.opencv.android.Utils;
import org.opencv.core.DMatch;
import org.opencv.core.Mat;
import org.opencv.core.MatOfDMatch;
import org.opencv.core.MatOfKeyPoint;
import org.opencv.features2d.DescriptorMatcher;
import org.opencv.features2d.SIFT;
import java.util.ArrayList;
import java.util.List;

public class SIFTTest {

  public static boolean isTemplateMatch(Context mContext, Bitmap srcBitmap) {

    float nndrRatio =0.7f;//邻近距离阀值,这里设置既定值为0.7,该值可自行调整

    BitmapFactory.Options options =new BitmapFactory.Options();

    options.inPreferredConfig = Bitmap.Config.ARGB_8888;

    Bitmap bmp1 = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.template1, options);

    Mat tempMat =new Mat();

    Utils.bitmapToMat(bmp1, tempMat);

    Mat srcMat =new Mat();

    Utils.bitmapToMat(srcBitmap, srcMat);

    MatOfKeyPoint templateKeyPoints =new MatOfKeyPoint();

    MatOfKeyPoint srcKeyPoints =new MatOfKeyPoint();

    //初始化SIFT
    SIFT siftDetector = SIFT.create();

    MatOfKeyPoint templateDescriptors =new MatOfKeyPoint();

    MatOfKeyPoint srcDescriptors =new MatOfKeyPoint();

    DescriptorMatcher descriptorMatcher = DescriptorMatcher.create(DescriptorMatcher.FLANNBASED);

    //获取模板图的特征点
    siftDetector.detect(tempMat, templateKeyPoints);

    siftDetector.detect(srcMat, srcKeyPoints);

    siftDetector.compute(tempMat, templateKeyPoints, templateDescriptors);

    siftDetector.compute(srcMat, srcKeyPoints, srcDescriptors);

    List matches =new ArrayList<>();

    //获取最佳匹配点列表
    descriptorMatcher.knnMatch(templateDescriptors, srcDescriptors, matches,2);

    int matchCount =0;

    for (int i =0; i < matches.size(); i++) {

        MatOfDMatch match = matches.get(i);

        DMatch[] array = match.toArray();

        DMatch m1 = array[0];

        DMatch m2 = array[1];

        //用邻近距离比值法(NDDR)计算匹配点数
        if (m1.distance <= m2.distance * nndrRatio) {

            ++matchCount;

        }

    }

    Log.i("###","======matchCount========" + matchCount);

    if (matchCount >=4) {

        //当匹配后的特征点大于等于 4 个,则认为模板图在原图中(这边匹配特征点个数可以根据实际情况自己设置)
        return true;

    }

    return false;

  }

}

OK,有轮子就是这么简单,这样就实现了用SIFT算法比较两张图片的相似度。因为这是我第一次使用,有可能会存在一些错误,如有发现问题请不吝赐教。

实际运行下来发现这个算法的精度的确挺高,果然名不虚传!但对我来说有一点不理想的地方就是运行速度偏慢,执行一次比较需要1~2s。如果单纯只是想比较两张静态图片(比如缩略图、原图的切片之类),大可不必用这么复杂的算法,比如下面这种直方图比较法,运行起来就快很多。

import android.content.Context;
import android.graphics.Bitmap;
import android.graphics.BitmapFactory;
import android.util.Log;
import com.wildma.idcardcamera.R;
import org.opencv.android.Utils;
import org.opencv.core.Mat;
import org.opencv.core.MatOfFloat;
import org.opencv.core.MatOfInt;
import org.opencv.imgproc.Imgproc;
import java.util.Arrays;

public class HistCompareTest {

public static boolean isTemplateMatch(Context mContext, Bitmap bitmap) {

    //读取模版图片
    BitmapFactory.Options options =new BitmapFactory.Options();

    options.inPreferredConfig = Bitmap.Config.ARGB_8888;

    Bitmap bmp1 = BitmapFactory.decodeResource(mContext.getResources(), R.mipmap.tag_circle, options);

    Mat mat1 =new Mat();

    Utils.bitmapToMat(bitmap, mat1);

    Mat mat2 =new Mat();

    Utils.bitmapToMat(bmp1, mat2);

    Mat mat_1 =new Mat();

    Mat mat_2 =new Mat();

    //颜色转换
    Imgproc.cvtColor(mat1, mat_1, Imgproc.COLOR_BGR2HSV);

    Imgproc.cvtColor(mat2, mat_2, Imgproc.COLOR_BGR2HSV);

    Mat hist_1 =new Mat();

    Mat hist_2 =new Mat();

    //颜色范围
    MatOfFloat ranges =new MatOfFloat(0f,256f);

    //直方图大小, 越大匹配越精确 (越慢)
    MatOfInt histSize =new MatOfInt(100);

    Imgproc.calcHist(Arrays.asList(mat_1),new MatOfInt(0),new Mat(), hist_1, histSize, ranges);

    Imgproc.calcHist(Arrays.asList(mat_2),new MatOfInt(0),new Mat(), hist_2, histSize, ranges);

    // 相似度系数,该值越接近1表示越相似
    double histVal = Imgproc.compareHist(hist_1, hist_2, Imgproc.CV_COMP_CORREL);

    Log.i("###","=====histVal======" + histVal);
    //相似度值大于0.6,则认为两张图片相似(该值可以根据你实际情况自由设置)

    if (histVal >0.6){
        return true;
    }

    return false;

  }

}

好了,这次的分享就到这里啦!希望对你会有所帮助·~

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,386评论 6 479
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,939评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,851评论 0 341
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,953评论 1 278
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,971评论 5 369
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,784评论 1 283
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,126评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,765评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 43,148评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,744评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,858评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,479评论 4 322
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,080评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,053评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,278评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,245评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,590评论 2 343