OpenCV 人脸识别、图片相似度检测

OpenCVCheck

项目地址:SOFTPOWER1991/OpenCVCheck 

简介: OpenCV 人脸识别、图片相似度检测

更多:作者   提 Bug   

标签:

 

OpenCV 人脸识别、图片相似度检测

检测任意两张图片的相似度思路

  1. 加载两张图片为 bitmap 进入内存
  2. 将内存中的两张图片 bitmap 转换为 Mat 矩阵(Mat 类是 OpenCV 最基本的一个数据类型,它可以表示一个多维的多通道的数组。Mat 常用来存储图像,包括单通道二维数组——灰度图,多通道二维数组——彩色图)
  3. 把 Mat 矩阵的 type 转换为 Cv_8uc1(1 通道 8 位矩阵)类型,然后转换为 Cv_32F, 因为在 c++代码中会判断他的类型。
  4. 通过 OpenCv 来进行俩个矩阵的比较(俩个矩阵必须一样大小的高宽)

识别图片中是否有人脸思路

  1. 需要一个人脸的 Haar 特征分类器就是一个 XML 文件,该文件中会描述人脸的 Haar 特征值,CascadeClassifier 人脸探测器将该特征值集合加载入内存
  2. 加载图片为 bitmap 进入内存,将 bitmap 转换为 Mat 矩阵。
  3. 有了 Mat 矩阵,然后通过调用 OpenCV 的 Native 方法,人脸探测器 CascadeClassifier 在该 Mat 矩阵中检测当前是否有人脸。
  4. 如果有,我们会获取到一个 Rect 数组,里面会有人脸数据,然后将人脸画在屏幕上,方框或者圆形

识别两张图片中的人脸是否是同一个人脸思路

  1. 识别出人脸后会得到两个人脸的 Rect 数组,然后比较这两个 Rect 数组的相似度即可!

实现步骤

工程目录准备

  1. 新建 Android Studio 项目 OpenCVCheck
  2. 导入 OpenCVLibrary320
  3. 在 module 下的 build.gradle 中引入 OpenCVLibrary 的编译:

      compile project(':openCVLibrary320')
    

检测任意两张图片的相似度的实现步骤

  1. 初始化 OpenCV:

     static {
         if (OpenCVLoader.initDebug()) {
             Log.e(TAG, "OpenCV load success !");
         } else {
             Log.e(TAG, "OpenCV load failed !");
         }
     }
    
  2. 加载两张图片进入内存

     Bitmap mBitmap1 = BitmapFactory.decodeResource(getResources(), R.mipmap.pic1);
     Bitmap mBitmap2 = BitmapFactory.decodeResource(getResources(), R.mipmap.pic2);
    
  3. 将内存中的两张图片 bitmap 转换为 Mat 矩阵

     Mat mat1 = new Mat();
     Mat mat2 = new Mat();
     Mat mat11 = new Mat();
     Mat mat22 = new Mat();
     Utils.bitmapToMat(mBitmap1, mat1);
     Utils.bitmapToMat(mBitmap2, mat2);
    
     Imgproc.cvtColor(mat1, mat11, Imgproc.COLOR_BGR2GRAY);
     Imgproc.cvtColor(mat2, mat22, Imgproc.COLOR_BGR2GRAY);
    
  4. 把 Mat 矩阵的 type 转换为 Cv_8uc1(1 通道 8 位矩阵)类型,然后转换为 Cv_32F,通过 OpenCV 来进行俩个矩阵的比较

     /**
      * 比较来个矩阵的相似度
      *
      * @param srcMat
      * @param desMat
      */
     public void comPareHist(Mat srcMat, Mat desMat) {
    
         srcMat.convertTo(srcMat, CvType.CV_32F);
         desMat.convertTo(desMat, CvType.CV_32F);
         double target = Imgproc.compareHist(srcMat, desMat, Imgproc.CV_COMP_CORREL);
    
         textView.setText("相似度:" + target);
     }
    

识别图片中是否有人脸步骤

  1. 初始化 OpenCV

     if (!OpenCVLoader.initDebug()) {
        Log.d(TAG, "Internal OpenCV library not found. Using OpenCV Manager for initialization");
        OpenCVLoader.initAsync(OpenCVLoader.OPENCV_VERSION_2_4_11, this, mLoaderCallback);
     } else {
        Log.d(TAG, "OpenCV library found inside package. Using it!");
        mLoaderCallback.onManagerConnected(LoaderCallbackInterface.SUCCESS);
     }
    
  2. 编译 .so 库

    通过 ndk 来编译 jni 文件下的.cpp 文件,生成.so 库,以备程序使用

  3. 加载.so 库

          // 在 Opencv 初始化完成后,调用 Native 库
          System.loadLibrary("detection_based_tracker");
    
  4. 加载需要的人脸的 Haar 特征分类器就是一个 XML 文件,该文件中会描述人脸的 Haar 特征值

    ``` private BaseLoaderCallback mLoaderCallback = new BaseLoaderCallback(this) {

        @Override
        public void onManagerConnected(int status) {
            switch (status) {
                case LoaderCallbackInterface.SUCCESS: {
                    Log.i(TAG, "OpenCV loaded successfully");
    
                    // Load native library after(!) OpenCV initialization
                    System.loadLibrary("detection_based_tracker");
    
                    try {
                        // load cascade file from application resources
                        InputStream is = getResources().openRawResource(R.raw.lbpcascade_frontalface);
                        File cascadeDir = getDir("cascade", Context.MODE_PRIVATE);
                        mCascadeFile = new File(cascadeDir, "lbpcascade_frontalface.xml");
                        FileOutputStream os = new FileOutputStream(mCascadeFile);
    
                        byte[] buffer = new byte[4096];
                        int bytesRead;
                        while ((bytesRead = is.read(buffer)) != -1) {
                            os.write(buffer, 0, bytesRead);
                        }
                        is.close();
                        os.close();
    
                        mJavaDetector = new CascadeClassifier(mCascadeFile.getAbsolutePath());
                        if (mJavaDetector.empty()) {
                            Log.e(TAG, "Failed to load cascade classifier");
                            mJavaDetector = null;
                        } else
                            Log.i(TAG, "Loaded cascade classifier from " + mCascadeFile.getAbsolutePath());
    
                        mNativeDetector = new DetectionBasedTracker(mCascadeFile.getAbsolutePath(), 0);
    
                        cascadeDir.delete();
    
                    } catch (IOException e) {
                        e.printStackTrace();
                        Log.e(TAG, "Failed to load cascade. Exception thrown: " + e);
                    }
    
               }
               break;
               default: {
                   super.onManagerConnected(status);
               }
               break;
           }
       }
   };
```
  1. 加载图片进入内存,得到 Mat 矩阵,有了 Mat 矩阵,然后通过调用 OpenCV 的 Native 方法,人脸探测器 CascadeClassifier 在该 Mat 矩阵中检测当前是否有人脸

              Bitmap imgtemp = BitmapFactory.decodeResource(getResources(), R.mipmap.twop);
    
              Utils.bitmapToMat(imgtemp, mRgba);
    
              Mat mat1 = new Mat();
    
              Utils.bitmapToMat(imgtemp, mat1);
    
              Imgproc.cvtColor(mat1, mGray, Imgproc.COLOR_BGR2GRAY);
    
              if (mAbsoluteFaceSize == 0) {
                  int height = mGray.rows();
                  if (Math.round(height * mRelativeFaceSize) > 0) {
                      mAbsoluteFaceSize = Math.round(height * mRelativeFaceSize);
                  }
                  mNativeDetector.setMinFaceSize(mAbsoluteFaceSize);
              }
    
              MatOfRect faces = new MatOfRect();
    
              if (mDetectorType == JAVA_DETECTOR) {
                  if (mJavaDetector != null)
                      mJavaDetector.detectMultiScale(mGray, faces, 1.1, 2, 2, // TODO: objdetect.CV_HAAR_SCALE_IMAGE
                              new Size(mAbsoluteFaceSize, mAbsoluteFaceSize), new Size());
              } else if (mDetectorType == NATIVE_DETECTOR) {
                  if (mNativeDetector != null)
                      mNativeDetector.detect(mGray, faces);
              } else {
                  Log.e(TAG, "Detection method is not selected!");
              }
    
  2. 如果有,我们会获取到一个 Rect 数组,里面会有人脸数据,然后将人脸画在屏幕上,方框或者圆形

      Rect[] facesArray = faces.toArray();
      for (int i = 0; i < facesArray.length; i++)
          Imgproc.rectangle(mRgba, facesArray[i].tl(), facesArray[i].br(), FACE_RECT_COLOR, 3);
    
      Utils.matToBitmap(mRgba, imgtemp, true);
    
      imageView.setImageBitmap(imgtemp);
    
## 识别两张图片中的人脸是否是同一个人脸步骤

 这个功能前面的步骤跟检测人脸的一样,唯一不同的就是:检测出两个人脸的 Rect 数组后,进行相似度比较:

 ```
 /**
      * 特征对比
      *
      * @param file1 人脸特征
      * @param file2 人脸特征
      * @return 相似度
      */
     public double CmpPic(String file1, String file2) {
         try {
             int l_bins = 256;
             int hist_size[] = {l_bins};
             float v_ranges[] = {0, 255};
             float ranges[][] = {v_ranges};
             opencv_core.IplImage Image1 = cvLoadImage(getFilePath(file1), CV_LOAD_IMAGE_GRAYSCALE);
             opencv_core.IplImage Image2 = cvLoadImage(getFilePath(file2), CV_LOAD_IMAGE_GRAYSCALE);
             opencv_core.IplImage imageArr1[] = {Image1};
             opencv_core.IplImage imageArr2[] = {Image2};
             opencv_imgproc.CvHistogram Histogram1 = opencv_imgproc.CvHistogram.create(1, hist_size, CV_HIST_ARRAY, ranges, 1);
             opencv_imgproc.CvHistogram Histogram2 = opencv_imgproc.CvHistogram.create(1, hist_size, CV_HIST_ARRAY, ranges, 1);
             cvCalcHist(imageArr1, Histogram1, 0, null);
             cvCalcHist(imageArr2, Histogram2, 0, null);
             cvNormalizeHist(Histogram1, 100.0);
             cvNormalizeHist(Histogram2, 100.0);

             double c1 = cvCompareHist(Histogram1, Histogram2, CV_COMP_CORREL) * 100;
             double c2 = cvCompareHist(Histogram1, Histogram2, CV_COMP_INTERSECT);
             return (c1 + c2) / 2;
         } catch (Exception e) {
             e.printStackTrace();
             return -1;
         }
     }
 ```
MICROSOFT 基础类库: 人脸相似度检测MFC 项目概述 =============================================================================== 应用程序向导已为您创建了这个 人脸相似度检测MFC 应用程序。此应用程序不仅演示 Microsoft 基础类的基本使用方法,还可作为您编写应用程序的起点。 本文件概要介绍组成 人脸相似度检测MFC 应用程序的每个文件的内容。 人脸相似度检测MFC.vcproj 这是使用应用程序向导生成的 VC++ 项目的主项目文件。 它包含生成该文件的 Visual C++ 的版本信息,以及有关使用应用程序向导选择的平台、配置和项目功能的信息。 人脸相似度检测MFC.h 这是应用程序的主要头文件。它包括其他项目特定的头文件(包括 Resource.h),并声明 C人脸相似度检测MFCApp 应用程序类。 人脸相似度检测MFC.cpp 这是包含应用程序类 C人脸相似度检测MFCApp 的主要应用程序源文件。 人脸相似度检测MFC.rc 这是程序使用的所有 Microsoft Windows 资源的列表。它包括 RES 子目录中存储的图标、位图和光标。此文件可以直接在 Microsoft Visual C++ 中进行编辑。项目资源位于 2052 中。 res\人脸相似度检测MFC.ico 这是用作应用程序图标的图标文件。此图标包括在主要资源文件 人脸相似度检测MFC.rc 中。 res\MFC.rc2 此文件包含不在 Microsoft Visual C++ 中进行编辑的资源。您应该将不可由资源编辑器编辑的所有资源放在此文件中。 ///////////////////////////////////////////////////////////////////////////// 应用程序向导创建一个对话框类: 人脸相似度检测MFCDlg.h,人脸相似度检测MFCDlg.cpp - 对话框 这些文件包含 C人脸相似度检测MFCDlg 类。该类定义应用程序主对话框的行为。该对话框的模板位于 人脸相似度检测MFC.rc 中,该文件可以在 Microsoft Visual C++ 中进行编辑。 ///////////////////////////////////////////////////////////////////////////// 其他功能: ActiveX 控件 应用程序包括对使用 ActiveX 控件的支持。 ///////////////////////////////////////////////////////////////////////////// 其他标准文件: StdAfx.h,StdAfx.cpp 这些文件用于生成名为 人脸相似度检测MFC.pch 的预编译头 (PCH) 文件和名为 StdAfx.obj 的预编译类型文件。 Resource.h 这是标准头文件,它定义新的资源 ID。 Microsoft Visual C++ 读取并更新此文件。 人脸相似度检测MFC.manifest 应用程序清单文件供 Windows XP 用来描述应用程序 对特定版本并行程序集的依赖性。加载程序使用此 信息从程序集缓存加载适当的程序集或 从应用程序加载私有信息。应用程序清单可能为了重新分发而作为 与应用程序可执行文件安装在相同文件夹中的外部 .manifest 文件包括, 也可能以资源的形式包括在该可执行文件中。 ///////////////////////////////////////////////////////////////////////////// 其他注释: 应用程序向导使用“TODO:”指示应添加或自定义的源代码部分。 如果应用程序在共享的 DLL 中使用 MFC,则需要重新发布这些 MFC DLL;如果应用程序所用的语言与操作系统的当前区域设置不同,则还需要重新发布对应的本地化资源 MFC90XXX.DLL。有关这两个主题的更多信息,请参见 MSDN 文档中有关 Redistributing Visual C++ applications (重新发布 Visual C++ 应用程序)的章节。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值