Android開發:即時處理網路攝影機預覽幀視頻——淺析PreviewCallback,onPreviewFrame,AsyncTask的綜合應用

來源:互聯網
上載者:User

很多時候,android網路攝影機模組不僅預覽,拍照這麼簡單,而是需要在預覽視頻的時候,能夠做出一些檢測,比如最常見的臉部偵測。在未按下拍照按鈕前,就檢測出人臉然後矩形框標示出來,再按拍照。那麼如何獲得預覽幀視頻嗎?

只需要在Activity裡繼承PreviewCallback這個介面就行了。樣本如下:

public class RectPhoto extends Activity implements SurfaceHolder.Callback, PreviewCallback{}。(注意這個SurfaceHolder.Callback是用來預覽網路攝影機視頻,參見我的前貼)。

繼承這個方法後,會自動重載這個函數:public void onPreviewFrame(byte[] data, Camera camera) {}這個函數裡的data就是即時預覽幀視頻。一旦程式調用PreviewCallback介面,就會自動調用onPreviewFrame這個函數。調用PreviewCallback的方法有三種,可以參考這裡,總共有三種方式調用這個回調。所謂回調就是當條件滿足時,自動觸發調用這個函數。分別是:.setPreviewCallback,
setOneShotPreviewCallback, setPreviewCallbackWithBuffer, 我一般是使用第二種方式。

       這裡解釋下,如果Activity繼承了PreviewCallback這個介面,只需                Camera.setOneShotPreviewCallback(this);就可以了。程式會自動調用主類Activity裡的onPreviewFrame函數。如果Camera.setOneShotPreviewCallback()這個函數是在主類Activity裡的內部類如class
A裡面,裡面的參數應寫為Camera.setOneShotPreviewCallback(YourActivity.this)。當然這裡,也可以定義一個變數,如Camera.PreviewCallback mPreviewCallback,在調用的時候用Camera.setOneShotPreviewCallback(mPreviewCallback)來完成。相信很多人都熟悉這點,就不羅嗦了。

      按理說只要在onPreviewFrame()這個函數裡寫你的處理常式就可以了。當通常不這麼做,因為處理即時預覽幀視頻的演算法可能比較複雜,這就需要藉助AsyncTask開啟一個線程在幕後處理資料。這裡假設我們定義一個FaceTask來進行臉部偵測,可以這樣寫:

    /*自訂的FaceTask類,開啟一個線程分析資料*/
    private class FaceTask extends AsyncTask<Void, Void, Void>{

        private byte[] mData;
        //建構函式
        PalmTask(byte[] data){
            this.mData = data;
        }
        
        @Override
        protected Void doInBackground(Void... params) {
            // TODO Auto-generated method stub
            Size size = myCamera.getParameters().getPreviewSize(); //擷取預覽大小
            final int w = size.width;  //寬度
            final int h = size.height;
            final YuvImage image = new YuvImage(mData, ImageFormat.NV21, w, h, null);
            ByteArrayOutputStream os = new ByteArrayOutputStream(mData.length);
            if(!image.compressToJpeg(new Rect(0, 0, w, h), 100, os)){
                return null;
            }
            byte[] tmp = os.toByteArray();
            Bitmap bmp = BitmapFactory.decodeByteArray(tmp, 0,tmp.length); 
           
doSomethingNeeded(bmp);   //自己定義的即時分析預覽幀視頻的演算法

          return null;
        }
        
    }  

注意上面的bmp就是Bitmap格式的即時預覽幀資料。doSomethingNeeded(bmp) 就是你要對預覽幀視頻進行的處理,可以是檢測人臉或其他,如分析有無火災。或者是進行傳輸。 
另外,這裡是通過YuvImage和ImageFormat.NV21來解析資料的。在華為u9200上,android4.0.3的系統運行良好。不同手機上支援的格式可能有所不同。網上也有自己寫演算法進行轉化的,需要的可以自己找,但這裡如果支援這個格式就不用自己寫轉換演算法了。 

     onPreviewFrame()裡可以這樣寫:

     /*擷取預覽幀視頻*/
    public void onPreviewFrame(byte[] data, Camera camera) {
        // TODO Auto-generated method stub
        if(null != mFaceTask){
            switch(mFaceTask.getStatus()){
            case RUNNING:
                return;
            case PENDING:
                mFaceTask.cancel(false);
                break;
            }
        }
        mFaceTask = new PalmTask(data);
        mFaceTask.execute((Void)null);
        
    }

    上面的mFaceTask是一個全域變數。通過onPreviewFrame,AsyncTask的綜合應用,讓複雜的處理演算法執行在後台,也就是doInBackground這裡,是不是比較綠色?

     接下來就是什麼時候觸發onPreviewFrame()這個函數裡,可以是按一個按鍵觸發一次,就在按鍵的監聽裡寫上       myCamera.setOneShotPreviewCallback(RectPhoto.this);便會自動觸發一次。有人說想先聚焦,然後再分析預覽幀。就在onAutofocus裡的回調寫。如下:

        //自動聚焦變數回調
        myAutoFocusCallback = new AutoFocusCallback() {

            public void onAutoFocus(boolean success, Camera camera) {
                // TODO Auto-generated method stub
                if(success)//success表示對焦成功
                {
                    Log.i(tag, "myAutoFocusCallback: success...");
                    myCamera.setOneShotPreviewCallback(RectPhoto.this);
                   

                }
                else
                {
                    //未對焦成功

                    Log.i(tag, "myAutoFocusCallback: 失敗了...");

                  //這裡也可以加上myCamera.autoFocus(myAutoFocusCallback),如果聚焦失敗就再次啟動聚焦。
                }

            }
        };

     大多數時候,希望程式自動每隔多長時間,自動進行一次檢測預覽幀。這也好辦,實施如下:

class ScanThread implements Runnable{public void run() {// TODO Auto-generated method stubwhile(!Thread.currentThread().isInterrupted()){try {if(null != myCamera && isPreview){    //myCamera.autoFocus(myAutoFocusCallback);myCamera.setOneShotPreviewCallback(RectPhoto.this);Log.i(tag, "setOneShotPreview...");}Thread.sleep(1500);} catch (InterruptedException e) {// TODO Auto-generated catch blocke.printStackTrace();Thread.currentThread().interrupt();}}}}

在onCreate裡new Thread(new ScanThread()).start()開啟掃描線程。如果想手動觸發中止這種掃描活動,可以在ScanThread裡的while迴圈裡設定標誌位,具體可看我以前的博文。


    最後提醒的是,如果程式中加入了previewCallback,在surfaceDestroy釋放camera的時候,最好執行myCamera.setOneShotPreviewCallback(null); 或者myCamera.setPreviewCallback(null);中止這種回調,然後再釋放camera更安全。否則可能會報錯。

      歡迎android愛好者加群248217350,備忘:yanzi

-----------------------------------------------------------------本文系原創,轉載請註明作者:yanzi1225627

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.