MipcaActivityCapture代码阅读-surfaceview的生命周期

前言

源代码来自http://blog.csdn.net/xiaanming/article/details/10163203,由于不懂surfaceDestroyed的调用时机,网上一般都说在surfaceview隐藏的时候调用,但是按back键、按home键、锁屏、跳转到另一个非完全覆盖的activity是否会调用surfaceDestroyed,不是很清楚。 所以用打印日志的方式分析了下各个函数的执行调用过程。
很多surfaceview的demo中都用到了hasSurface,但是hasSurface又有什么作用呢?曾经深表怀疑,仔细分析过代码才明白
四种方式的日志

back

进入MipcaActivityCapture这个activity
01-09 16:59:51.118:E/MipcaActivityCapture(25152): onCreate
01-09 16:59:51.141:E/MipcaActivityCapture(25152): onResume
01-09 16:59:51.141:E/MipcaActivityCapture(25152): hasSurface=false
01-09 16:59:51.259:E/MipcaActivityCapture(25152): surfaceCreated
01-09 16:59:51.259:E/MipcaActivityCapture(25152): initCamera

点back

01-09 16:59:54.102:E/MipcaActivityCapture(25152): onPause
01-09 16:59:54.227:E/MipcaActivityCapture(25152): surfaceDestroyed
01-09 16:59:54.391:E/MipcaActivityCapture(25152): onStop
01-09 16:59:54.399:E/MipcaActivityCapture(25152): onDestroy

home

进入MipcaActivityCapture这个activity
01-09 16:59:51.118:E/MipcaActivityCapture(25152): onCreate
01-09 16:59:51.141:E/MipcaActivityCapture(25152): onResume
01-09 16:59:51.141:E/MipcaActivityCapture(25152): hasSurface=false
01-09 16:59:51.259:E/MipcaActivityCapture(25152): surfaceCreated
01-09 16:59:51.259:E/MipcaActivityCapture(25152): initCamera

点home

01-09 17:28:47.196:E/MipcaActivityCapture(26547): onPause
01-09 17:28:47.321:E/MipcaActivityCapture(26547): surfaceDestroyed
01-09 17:28:47.704:E/MipcaActivityCapture(26547): onStop

锁屏

进入MipcaActivityCapture这个activity
01-09 17:31:03.259:E/MipcaActivityCapture(26655): onCreate
01-09 17:31:03.282:E/MipcaActivityCapture(26655): hasSurface=false
01-09 17:31:03.282:E/MipcaActivityCapture(26655): onResume
01-09 17:31:03.438:E/MipcaActivityCapture(26655): surfaceCreated
01-09 17:31:03.438:E/MipcaActivityCapture(26655): initCamera

锁屏(按关机键)

01-09 17:31:09.907:E/MipcaActivityCapture(26655): onPause
01-09 17:31:09.938:E/MipcaActivityCapture(26655): onStop

解锁(按关机键)

01-09 17:35:51.509:E/MipcaActivityCapture(27383): onResume
01-09 17:35:51.509:E/MipcaActivityCapture(27383): hasSurface=true
01-09 17:35:51.509:E/MipcaActivityCapture(27383): initCamera

跳转到一个非完全覆盖的activity

进入MipcaActivityCapture这个activity
01-09 17:43:38.126:E/MipcaActivityCapture(27383): onCreate
01-09 17:43:38.165:E/MipcaActivityCapture(27383): onResume
01-09 17:43:38.165:E/MipcaActivityCapture(27383): hasSurface=false
01-09 17:43:38.251:E/MipcaActivityCapture(27383): surfaceCreated
01-09 17:43:38.251:E/MipcaActivityCapture(27383): initCamera

跳转到一个非完全覆盖的activity时

01-09 17:43:44.446:E/MipcaActivityCapture(27383): onPause
01-09 17:43:44.477: E/OtherActivity(27383):---------------onCreate--------------------
01-09 17:43:44.477: E/OtherActivity(27383):---------------onStart--------------------
01-09 17:43:44.477: E/OtherActivity(27383):---------------onResume--------------------

按back

01-09 17:43:50.954: E/OtherActivity(27383):---------------onPause--------------------
01-09 17:43:50.977:E/MipcaActivityCapture(27383): onResume
01-09 17:43:50.977: E/MipcaActivityCapture(27383):hasSurface=true
01-09 17:43:50.977:E/MipcaActivityCapture(27383): initCamera
01-09 17:43:51.954: E/OtherActivity(27383):---------------onStop--------------------
01-09 17:43:51.954: E/OtherActivity(27383):---------------onDestroy--------------------

总结

onPause会调用CameraManager.get().closeDriver();
surfaceCreated一般在OnResume之后调用,
按back或home都会调用surfaceDestroyed ,surfaceDestroyed在OnPause和OnStop之间调用。
但是锁屏不会调用surfaceDestroyed,解锁也不会调surfaceCreated
跳转到一个非完全覆盖的activity时也不会调surfaceDestroyed,返回时也不会调surfaceCreated

按back/home或者锁屏都会调用OnPause,导致CameraManager.get().closeDriver();所以下次进入activity的时候要initCamera,这是怎么实现的呢?看下面这段代码,展现了hasSurface的作用。

        if(hasSurface) {
            initCamera(surfaceHolder);
        }else {
            surfaceHolder.addCallback(this);
            surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);
        }

锁屏解锁后hasSurface为true,调用initCamera。
按back或者home再回来时hasSurface为false,此时在onResume之后会调用surfaceCreated,在surfaceCreated里面会调用initCamera。
代码
稍微改了下原代码

view plain copy

派生到我的代码片
派生到我的代码片

package com.example.qr_codescan;  
  
import java.io.IOException;  
import java.util.Vector;  
  
import android.app.Activity;  
import android.app.AlertDialog;  
import android.app.Dialog;  
import android.content.Intent;  
import android.content.res.AssetFileDescriptor;  
import android.graphics.Bitmap;  
import android.media.AudioManager;  
import android.media.MediaPlayer;  
import android.media.MediaPlayer.OnCompletionListener;  
import android.os.Bundle;  
import android.os.Handler;  
import android.os.Vibrator;  
import android.util.Log;  
import android.view.SurfaceHolder;  
import android.view.SurfaceHolder.Callback;  
import android.view.SurfaceView;  
import android.view.View;  
import android.view.View.OnClickListener;  
import android.widget.Button;  
import android.widget.Toast;  
  
import com.google.zxing.BarcodeFormat;  
import com.google.zxing.Result;  
import com.mining.app.zxing.camera.CameraManager;  
import com.mining.app.zxing.decoding.CaptureActivityHandler;  
import com.mining.app.zxing.decoding.InactivityTimer;  
import com.mining.app.zxing.view.ViewfinderView;  
/** 
 * Initial the camera 
 * @author Ryan.Tang 
 */  
public class MipcaActivityCapture extends Activity implements Callback {  
  
    private static final String TAG = MipcaActivityCapture.class.getSimpleName();  
    private CaptureActivityHandler handler;  
    private ViewfinderView viewfinderView;  
    private boolean hasSurface;  
    private Vector<BarcodeFormat> decodeFormats;  
    private String characterSet;  
    private InactivityTimer inactivityTimer;  
    private MediaPlayer mediaPlayer;  
    private boolean playBeep;  
    private static final float BEEP_VOLUME = 0.10f;  
    private boolean vibrate;  
    Button btn;  
  
    /** Called when the activity is first created. */  
    @Override  
    public void onCreate(Bundle savedInstanceState) {  
        Log.e(TAG, "onCreate");  
        super.onCreate(savedInstanceState);  
        setContentView(R.layout.activity_capture1);  
        //ViewUtil.addTopView(getApplicationContext(), this, R.string.scan_card);  
        CameraManager.init(getApplication());  
        viewfinderView = (ViewfinderView) findViewById(R.id.viewfinder_view);  
          
        Button mButtonBack = (Button) findViewById(R.id.button_back);  
        mButtonBack.setOnClickListener(new OnClickListener() {  
              
            @Override  
            public void onClick(View v) {  
                MipcaActivityCapture.this.finish();  
                  
            }  
        });  
        hasSurface = false;  
        inactivityTimer = new InactivityTimer(this);  
          
        btn=(Button) findViewById(R.id.btn1);  
        btn.setOnClickListener(new OnClickListener() {  
              
            @Override  
            public void onClick(View v) {  
  
                Intent intent=new Intent(MipcaActivityCapture.this, OtherActivity.class);  
                startActivity(intent);  
                  
            }  
        });  
    }  
  
    @Override  
    protected void onResume() {  
        Log.e(TAG, "onResume");  
        super.onResume();  
        SurfaceView surfaceView = (SurfaceView) findViewById(R.id.preview_view);  
        SurfaceHolder surfaceHolder = surfaceView.getHolder();  
        Log.e(TAG, "hasSurface="+hasSurface);  
        if (hasSurface) {  
            initCamera(surfaceHolder);  
        } else {  
            surfaceHolder.addCallback(this);  
            surfaceHolder.setType(SurfaceHolder.SURFACE_TYPE_PUSH_BUFFERS);  
        }  
        decodeFormats = null;  
        characterSet = null;  
  
        playBeep = true;  
        AudioManager audioService = (AudioManager) getSystemService(AUDIO_SERVICE);  
        if (audioService.getRingerMode() != AudioManager.RINGER_MODE_NORMAL) {  
            playBeep = false;  
        }  
        initBeepSound();  
        vibrate = true;  
   
          
    }  
  
    @Override  
    protected void onPause() {  
        super.onPause();  
        if (handler != null) {  
            handler.quitSynchronously();  
            handler = null;  
        }  
        CameraManager.get().closeDriver();  
  
        Log.e(TAG, "onPause");  
    }  
  
    @Override  
    protected void onStop() {  
        // TODO Auto-generated method stub  
        super.onStop();  
        Log.e(TAG, "onStop");  
    }  
    @Override  
    protected void onDestroy() {  
        inactivityTimer.shutdown();  
        super.onDestroy();  
        Log.e(TAG, "onDestroy");  
    }  
      
    /** 
     * 处理扫描结果 
     * @param result 
     * @param barcode 
     */  
    public void handleDecode(Result result, Bitmap barcode) {  
        inactivityTimer.onActivity();  
        playBeepSoundAndVibrate();  
        String resultString = result.getText();  
        if (resultString.equals("")) {  
            Toast.makeText(MipcaActivityCapture.this, "Scan failed!", Toast.LENGTH_SHORT).show();  
        }else {  
            Intent resultIntent = new Intent();  
            Bundle bundle = new Bundle();  
            bundle.putString("result", resultString);  
            bundle.putParcelable("bitmap", barcode);  
            resultIntent.putExtras(bundle);  
            this.setResult(RESULT_OK, resultIntent);  
        }  
        MipcaActivityCapture.this.finish();  
    }  
      
    private void initCamera(SurfaceHolder surfaceHolder) {  
  
        Log.e(TAG, "initCamera");  
        try {  
            CameraManager.get().openDriver(surfaceHolder);  
        } catch (IOException ioe) {  
            return;  
        } catch (RuntimeException e) {  
            return;  
        }  
        if (handler == null) {  
            handler = new CaptureActivityHandler(this, decodeFormats,  
                    characterSet);  
        }  
    }  
  
    @Override  
    public void surfaceChanged(SurfaceHolder holder, int format, int width,  
            int height) {  
  
    }  
  
    @Override  
    public void surfaceCreated(SurfaceHolder holder) {  
  
        Log.e(TAG, "surfaceCreated");  
        if (!hasSurface) {  
            hasSurface = true;  
            initCamera(holder);  
        }  
  
    }  
  
    @Override  
    public void surfaceDestroyed(SurfaceHolder holder) {  
        hasSurface = false;  
  
        Log.e(TAG, "surfaceDestroyed");  
  
    }  
  
    public ViewfinderView getViewfinderView() {  
        return viewfinderView;  
    }  
  
    public Handler getHandler() {  
        return handler;  
    }  
  
    public void drawViewfinder() {  
        viewfinderView.drawViewfinder();  
  
    }  
  
    private void initBeepSound() {  
        if (playBeep && mediaPlayer == null) {  
            // The volume on STREAM_SYSTEM is not adjustable, and users found it  
            // too loud,  
            // so we now play on the music stream.  
            setVolumeControlStream(AudioManager.STREAM_MUSIC);  
            mediaPlayer = new MediaPlayer();  
            mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);  
            mediaPlayer.setOnCompletionListener(beepListener);  
  
            AssetFileDescriptor file = getResources().openRawResourceFd(  
                    R.raw.beep);  
            try {  
                mediaPlayer.setDataSource(file.getFileDescriptor(),  
                        file.getStartOffset(), file.getLength());  
                file.close();  
                mediaPlayer.setVolume(BEEP_VOLUME, BEEP_VOLUME);  
                mediaPlayer.prepare();  
            } catch (IOException e) {  
                mediaPlayer = null;  
            }  
        }  
    }  
  
    private static final long VIBRATE_DURATION = 200L;  
  
    private void playBeepSoundAndVibrate() {  
        if (playBeep && mediaPlayer != null) {  
            mediaPlayer.start();  
        }  
        if (vibrate) {  
            Vibrator vibrator = (Vibrator) getSystemService(VIBRATOR_SERVICE);  
            vibrator.vibrate(VIBRATE_DURATION);  
        }  
    }  
  
    /** 
     * When the beep has finished playing, rewind to queue up another one. 
     */  
    private final OnCompletionListener beepListener = new OnCompletionListener() {  
        public void onCompletion(MediaPlayer mediaPlayer) {  
            mediaPlayer.seekTo(0);  
        }  
    };  
}  
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,098评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,213评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 149,960评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,519评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,512评论 5 364
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,533评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,914评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,574评论 0 256
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,804评论 1 296
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,563评论 2 319
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,644评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,350评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,933评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,908评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,146评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,847评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,361评论 2 342

推荐阅读更多精彩内容

  • 曾经有一份美好的爱情放在我的面前我没有珍惜。等到失去后才后悔莫及。如果可以再对小李说。毛欣想说。这辈子无缘再牵手。...
    毛欣与小李阅读 2,559评论 0 13
  • 夏日的傍晚 你盯着一面镜子 那里有一双忧郁的眼神 是百无聊懒后的清澈 门前的广场 两个小孩并排走过望了你一眼 你的...
    一尘九九阅读 192评论 2 0
  • 20年前,那时大学刚刚毕业,闲暇时间,我不去逛街购物,对衣服、包包、化妆品不感兴趣,唯一吸引我的是,去电脑市场“欣...
    V腾飞阅读 497评论 9 6
  • 连续灵修128天【经文】他祈祷耶和华,耶和华就允准他的祈求,垂听他的祷告,使他归回耶路撒冷,仍坐国位。玛拿西这才知...
    喜乐付阅读 4,871评论 0 0
  • 吟诗, 毋须模仿、毋须创作, 吟诗只是一种生活。 唱词, 毋须刻意、毋须雕琢, 唱词只是一腔烈火。 每天都是一首新...
    羽扇闲人阅读 190评论 1 0