Android中的人臉檢測(靜態和動態)

(1)背景。

       Google 於2006年8月收購Neven Vision 公司 (該公司擁有10多項應用於移動設備領域的圖像識別的專利),以此獲得了圖像識別的技術,並加入到android中。Android 中的人臉識別技術,用到的底層庫:android/external/neven/,framework 層:frameworks/base/media/java/android/media/FaceDetector.java。

       Java 層接口的限制:A,只能接受Bitmap 格式的數據;B,只能識別雙眼距離大於20 像素的人臉像(當然,這個可在framework層中修改);C,只能檢測出人臉的位置(雙眼的中心點及距離),不能對人臉進行匹配(查找指定的臉譜)。

        人臉識別技術的應用:A,爲Camera 添加人臉識別的功能,使得Camera 的取景器上能標識出人臉範圍;如果硬件支持,可以對人臉進行對焦。B,爲相冊程序添加按人臉索引相冊的功能,按人臉索引相冊,按人臉分組,搜索相冊。

(2)Neven庫給上層提供的主要方法:
        A,android.media.FaceDetector .FaceDetector(int width, int height, int maxFaces):Creates a FaceDetector, configured with the size of the images to be analysed and the maximum number of faces that can be detected. These parameters cannot be changed once the object is constructed.

        B,int android.media.FaceDetector .findFaces(Bitmap bitmap, Face [] faces):Finds all the faces found in a given Bitmap . The supplied array is populated with FaceDetector.Face s for each face found. The bitmap must be in 565 format (for now).

(3) 靜態圖片處理代碼實例:

       通過對位圖的處理,捕獲位圖中的人臉,並以綠框顯示,有多個人臉就提示多個綠框。首先新建一個activity,由於位圖資源會用代碼顯示出來,所以不需在layout中使用widget。

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
 
  1. package com.example.mydetect2;  
  2.   
  3. import android.os.Bundle;  
  4. import android.app.Activity;  
  5. import android.util.Log;  
  6. import android.view.Menu;  
  7. import android.content.Context;   
  8. import android.graphics.Bitmap;   
  9. import android.graphics.BitmapFactory;   
  10. import android.graphics.Canvas;   
  11. import android.graphics.Color;   
  12. import android.graphics.Paint;   
  13. import android.graphics.PointF;   
  14. import android.media.FaceDetector;  //人臉識別的關鍵類  
  15. import android.media.FaceDetector.Face;   
  16. import android.view.View;   
  17.   
  18. public class MainActivity2 extends Activity {  
  19.   
  20.     @Override  
  21.     protected void onCreate(Bundle savedInstanceState) {  
  22.         super.onCreate(savedInstanceState);  
  23.         //setContentView(R.layout.activity_main_activity2);  
  24.         setContentView(new myView(this));   //使用自建的view來顯示  
  25.         Log.i("zhangcheng","MainActivity2 run here");  
  26.     }  
  27.   
  28.     private class myView extends View{  
  29.         private int imageWidth, imageHeight;  
  30.         private int numberOfFace = 5;       //最大檢測的人臉數  
  31.         private FaceDetector myFaceDetect;  //人臉識別類的實例  
  32.         private FaceDetector.Face[] myFace; //存儲多張人臉的數組變量  
  33.         float myEyesDistance;           //兩眼之間的距離  
  34.         int numberOfFaceDetected;       //實際檢測到的人臉數  
  35.         Bitmap myBitmap;  
  36.   
  37.         public myView(Context context){     //view類的構造函數,必須有  
  38.             super(context);   
  39.             BitmapFactory.Options BitmapFactoryOptionsbfo = new BitmapFactory.Options();   
  40.             BitmapFactoryOptionsbfo.inPreferredConfig = Bitmap.Config.RGB_565;  //構造位圖生成的參數,必須爲565。類名+enum  
  41.             myBitmap = BitmapFactory.decodeResource(getResources(), R.drawable.baby, BitmapFactoryOptionsbfo);     
  42.             imageWidth = myBitmap.getWidth();   
  43.             imageHeight = myBitmap.getHeight();   
  44.             myFace = new FaceDetector.Face[numberOfFace];       //分配人臉數組空間  
  45.             myFaceDetect = new FaceDetector(imageWidth, imageHeight, numberOfFace);   
  46.             numberOfFaceDetected = myFaceDetect.findFaces(myBitmap, myFace);    //FaceDetector 構造實例並解析人臉  
  47.             Log.i("zhangcheng","numberOfFaceDetected is " + numberOfFaceDetected);  
  48.         }  
  49.           
  50.         protected void onDraw(Canvas canvas){           //override函數,必有  
  51.             canvas.drawBitmap(myBitmap, 00null);    //畫出位圖   
  52.             Paint myPaint = new Paint();   
  53.             myPaint.setColor(Color.GREEN);   
  54.             myPaint.setStyle(Paint.Style.STROKE);   
  55.             myPaint.setStrokeWidth(3);          //設置位圖上paint操作的參數  
  56.   
  57.             for(int i=0; i < numberOfFaceDetected; i++){  
  58.                 Face face = myFace[i];  
  59.                 PointF myMidPoint = new PointF();   
  60.                 face.getMidPoint(myMidPoint);   
  61.                 myEyesDistance = face.eyesDistance();   //得到人臉中心點和眼間距離參數,並對每個人臉進行畫框  
  62.                 canvas.drawRect(            //矩形框的位置參數  
  63.                         (int)(myMidPoint.x - myEyesDistance),   
  64.                         (int)(myMidPoint.y - myEyesDistance),   
  65.                         (int)(myMidPoint.x + myEyesDistance),   
  66.                         (int)(myMidPoint.y + myEyesDistance),   
  67.                         myPaint);  
  68.             }  
  69.         }  
  70.     }  
  71. }  


      以上爲activity,工程的xml文件沒有什麼特殊地方。最後得到的結果如下,圖片資源是png的也可以。

 


 (4) 動態預覽識別人臉代碼實例

       該過程用於後臺工作,沒有界面也沒有預覽。所以沒有采用上面那種處理位圖資源的方式。Import的類就不列出了,核心的代碼和流程如下:

A,打開攝像頭,獲得初步攝像頭回調數據,用到是setpreviewcallback

protected Camera mCameraDevice = null;// 攝像頭對象實例

private long mScanBeginTime = 0;   // 掃描開始時間
private long mScanEndTime = 0;   // 掃描結束時間
private long mSpecPreviewTime = 0;   // 掃描持續時間

private int orientionOfCamera ;   //前置攝像頭layout角度

int numberOfFaceDetected;    //最終識別人臉數目

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
 
  1. public void startFaceDetection() {  
  2.         try {  
  3.                 mCameraDevice = Camera.open(1);     //打開前置  
  4.                 if (mCameraDevice != null)  
  5.                     Log.i(TAG, "open cameradevice success! ");  
  6.             } catch (Exception e) {             //Exception代替很多具體的異常  
  7.                 mCameraDevice = null;  
  8.                 Log.w(TAG, "open cameraFail");  
  9.                 mHandler.postDelayed(r,5000);   //如果攝像頭被佔用,人眼識別每5秒檢測看有沒有釋放前置  
  10.                 return;  
  11.         }   
  12.               
  13.         Log.i(TAG, "startFaceDetection");  
  14.         Camera.Parameters parameters = mCameraDevice.getParameters();  
  15.         setCameraDisplayOrientation(1,mCameraDevice);              //設置預覽方向   
  16.               
  17.         mCameraDevice.setPreviewCallback(new PreviewCallback(){  
  18.             public void onPreviewFrame(byte[] data, Camera camera){  
  19.                 mScanEndTime = System.currentTimeMillis();   //記錄攝像頭返回數據的時間  
  20.                 mSpecPreviewTime = mScanEndTime - mScanBeginTime;  //從onPreviewFrame獲取攝像頭數據的時間  
  21.                 Log.i(TAG, "onPreviewFrame and mSpecPreviewTime = " + String.valueOf(mSpecPreviewTime));  
  22.                 Camera.Size localSize = camera.getParameters().getPreviewSize();  //獲得預覽分辨率  
  23.                 YuvImage localYuvImage = new YuvImage(data, 17, localSize.width, localSize.height, null);  
  24.                 ByteArrayOutputStream localByteArrayOutputStream = new ByteArrayOutputStream();  
  25.                 localYuvImage.compressToJpeg(new Rect(00, localSize.width, localSize.height), 80, localByteArrayOutputStream);    //把攝像頭回調數據轉成YUV,再按圖像尺寸壓縮成JPEG,從輸出流中轉成數組  
  26.                 byte[] arrayOfByte = localByteArrayOutputStream.toByteArray();  
  27.                 CameraRelease();   //及早釋放camera資源,避免影響camera設備的正常調用  
  28.                 StoreByteImage(arrayOfByte);  
  29.             }  
  30.         });  
  31.   
  32.         mCameraDevice.startPreview();         //該語句可放在回調後面,當執行到這裏,調用前面的setPreviewCallback  
  33.         mScanBeginTime = System.currentTimeMillis();// 記錄下系統開始掃描的時間  
  34.     }  

B,設置預覽方向的函數說明,該函數比較重要,因爲方向直接影響bitmap構造時的矩陣旋轉角度,影響最終人臉識別的成功與否

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
 
  1. public void setCameraDisplayOrientation(int paramInt, Camera paramCamera){  
  2.         CameraInfo info = new CameraInfo();  
  3.         Camera.getCameraInfo(paramInt, info);  
  4.         int rotation = ((WindowManager)getSystemService("window")).getDefaultDisplay().getRotation();  //獲得顯示器件角度  
  5.         int degrees = 0;  
  6.         Log.i(TAG,"getRotation's rotation is " + String.valueOf(rotation));  
  7.         switch (rotation) {  
  8.             case Surface.ROTATION_0: degrees = 0break;  
  9.             case Surface.ROTATION_90: degrees = 90break;  
  10.             case Surface.ROTATION_180: degrees = 180break;  
  11.             case Surface.ROTATION_270: degrees = 270break;  
  12.         }  
  13.   
  14.         orientionOfCamera = info.orientation;      //獲得攝像頭的安裝旋轉角度  
  15.         int result;  
  16.         if (info.facing == Camera.CameraInfo.CAMERA_FACING_FRONT) {  
  17.             result = (info.orientation + degrees) % 360;  
  18.             result = (360 - result) % 360;  // compensate the mirror  
  19.         } else {  // back-facing  
  20.             result = (info.orientation - degrees + 360) % 360;  
  21.         }  
  22.         paramCamera.setDisplayOrientation(result);  //注意前後置的處理,前置是映象畫面,該段是SDK文檔的標準DEMO  
  23.     }  

C,對攝像頭回調數據進行轉換並最終解成BITMAP後再人臉識別的過程

[java] view plaincopy在CODE上查看代碼片派生到我的代碼片
 
  1. public void StoreByteImage(byte[] paramArrayOfByte){  
  2.         mSpecStopTime = System.currentTimeMillis();  
  3.         mSpecCameraTime = mSpecStopTime - mScanBeginTime;  
  4.                        
  5.         Log.i(TAG, "StoreByteImage and mSpecCameraTime is " + String.valueOf(mSpecCameraTime));  
  6.   
  7.         BitmapFactory.Options localOptions = new BitmapFactory.Options();  
  8.             Bitmap localBitmap1 = BitmapFactory.decodeByteArray(paramArrayOfByte, 0, paramArrayOfByte.length, localOptions);  
  9.             int i = localBitmap1.getWidth();  
  10.             int j = localBitmap1.getHeight();   //從上步解出的JPEG數組中接出BMP,即RAW->JPEG->BMP  
  11.             Matrix localMatrix = new Matrix();  
  12.             //int k = cameraResOr;  
  13.             Bitmap localBitmap2 = null;  
  14.             FaceDetector localFaceDetector = null;  
  15.   
  16.         switch(orientionOfCamera){   //根據前置安裝旋轉的角度來重新構造BMP  
  17.             case 0:  
  18.                 localFaceDetector = new FaceDetector(i, j, 1);  
  19.                         localMatrix.postRotate(0.0F, i / 2, j / 2);  
  20.                         localBitmap2 = Bitmap.createBitmap(i, j, Bitmap.Config.RGB_565);  
  21.                 break;  
  22.             case 90:  
  23.                 localFaceDetector = new FaceDetector(j, i, 1);   //長寬互換  
  24.                         localMatrix.postRotate(-270.0F, j / 2, i / 2);  //正90度的話就反方向轉270度,一樣效果  
  25.                         localBitmap2 = Bitmap.createBitmap(i, j, Bitmap.Config.RGB_565);  
  26.                 break;                        
  27.             case 180:  
  28.                 localFaceDetector = new FaceDetector(i, j, 1);  
  29.                         localMatrix.postRotate(-180.0F, i / 2, j / 2);  
  30.                         localBitmap2 = Bitmap.createBitmap(i, j, Bitmap.Config.RGB_565);  
  31.                 break;  
  32.             case 270:  
  33.                 localFaceDetector = new FaceDetector(j, i, 1);  
  34.                         localMatrix.postRotate(-90.0F, j / 2, i / 2);  
  35.                         localBitmap2 = Bitmap.createBitmap(j, i, Bitmap.Config.RGB_565);  //localBitmap2應是沒有數據的  
  36.                 break;  
  37.         }  
  38.   
  39.         FaceDetector.Face[] arrayOfFace = new FaceDetector.Face[1];  
  40.             Paint localPaint1 = new Paint();  
  41.             Paint localPaint2 = new Paint();  
  42.         localPaint1.setDither(true);  
  43.             localPaint2.setColor(-65536);  
  44.             localPaint2.setStyle(Paint.Style.STROKE);  
  45.             localPaint2.setStrokeWidth(2.0F);  
  46.             Canvas localCanvas = new Canvas();  
  47.             localCanvas.setBitmap(localBitmap2);  
  48.             localCanvas.setMatrix(localMatrix);  
  49.             localCanvas.drawBitmap(localBitmap1, 0.0F, 0.0F, localPaint1); //該處將localBitmap1和localBitmap2關聯(可不要?)  
  50.   
  51.         numberOfFaceDetected = localFaceDetector.findFaces(localBitmap2, arrayOfFace); //返回識臉的結果  
  52.             localBitmap2.recycle();  
  53.             localBitmap1.recycle();   //釋放位圖資源  
  54.   
  55.         FaceDetectDeal(numberOfFaceDetected);  
  56.     }  

 

 

參考原文:http://blog.renren.com/share/234025064/6227088558

參考原文:http://www.csdn.net/article/a/2012-04-27/2805150

參考原文:http://www.cnblogs.com/feisky/archive/2010/09/12/1824320.html

 

轉自:http://blog.csdn.net/zhandoushi1982/article/details/8613916

發佈了340 篇原創文章 · 獲贊 4 · 訪問量 2萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章