Android SurfaceView的生命周期

来源:http://www.cnblogs.com/wuyudong/archive/2016/09/08/5851156.html
-Advertisement-
Play Games

本文利用SurfaceView來實現視頻的播放 本文地址:http://www.cnblogs.com/wuyudong/p/5851156.html,轉載請註明源地址。 在main.xml佈局文件添加用於視頻畫面繪製的SurfaceView 控制項: 項目佈局設計: 大部分代碼和《Android 多 ...


本文利用SurfaceView來實現視頻的播放

本文地址:http://www.cnblogs.com/wuyudong/p/5851156.html,轉載請註明源地址。

在main.xml佈局文件添加用於視頻畫面繪製的SurfaceView 控制項:

<SurfaceView android:layout_width="fill_parent" android:layout_height="240dip" android:id="@+id/surfaceView" />

項目佈局設計:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    xmlns:tools="http://schemas.android.com/tools"
    android:layout_width="match_parent"
    android:layout_height="match_parent"
    android:orientation="vertical"
    tools:context=".MainActivity" >

    <EditText
        android:id="@+id/et_path"
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:hint="請輸入視頻文件的路徑" />

    <LinearLayout
        android:layout_width="fill_parent"
        android:layout_height="wrap_content"
        android:orientation="horizontal" >

        <Button
            android:id="@+id/bt_play"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="播放" />

        <Button
            android:id="@+id/bt_pause"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="暫停" />

        <Button
            android:id="@+id/bt_replay"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="重播" />

        <Button
            android:id="@+id/bt_stop"
            android:layout_width="0dip"
            android:layout_height="wrap_content"
            android:layout_weight="1"
            android:text="停止" />
    </LinearLayout>
    
    <SurfaceView 
        android:id="@+id/sv"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        />

</LinearLayout>

大部分代碼和《Android 多媒體播放API簡介》中的完全一樣,修改一點點代碼即可:

public class MainActivity extends Activity implements OnClickListener {

    private EditText et_path;
    private Button bt_play, bt_replay, bt_pause, bt_stop;
    private SurfaceView sv;

    private MediaPlayer mediaPlayer;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        
        sv = (SurfaceView)findViewById(R.id.sv);
        et_path = (EditText) findViewById(R.id.et_path);
        bt_play = (Button) findViewById(R.id.bt_play);
        bt_replay = (Button) findViewById(R.id.bt_replay);
        bt_pause = (Button) findViewById(R.id.bt_pause);
        bt_stop = (Button) findViewById(R.id.bt_stop);

        bt_pause.setOnClickListener(this);
        bt_play.setOnClickListener(this);
        bt_replay.setOnClickListener(this);
        bt_stop.setOnClickListener(this);

    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
        case R.id.bt_play:
            play();
            break;
        case R.id.bt_replay:
            replay();
            break;
        case R.id.bt_stop:
            stop();
            break;
        case R.id.bt_pause:
            pause();
            break;
        default:
            break;
        }
    }

    /**
     * 暫停音樂
     */
    private void pause() {
        if ("繼續".equals(bt_pause.getText().toString().trim())) {
            mediaPlayer.start();
            bt_pause.setText("暫停");
            return;
        }
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.pause();
            bt_pause.setText("繼續");
            return;
        }
    }

    /**
     * 重新播放
     */
    private void replay() {
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.seekTo(0);
            return;
        }
        play();
    }

    /**
     * 停止播放音樂
     */
    private void stop() {
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.stop();
            mediaPlayer.release(); // 記得釋放資源
            mediaPlayer = null;
            bt_play.setEnabled(true);
        }
    }

    /**
     * 播放音樂
     */
    private void play() {
        String path = et_path.getText().toString().trim();
        File file = new File(path);
        if (file.exists() && file.length() > 0) {
            try {
                mediaPlayer = new MediaPlayer();
                mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);
                
                /* 設置Video影片以SurfaceHolder播放 */
                mediaPlayer.setDisplay(sv.getHolder());
                
                mediaPlayer.setDataSource(path);
                mediaPlayer.prepare(); // might take long! (for buffering, etc)
                mediaPlayer.start();
                mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
                    @Override
                    public void onCompletion(MediaPlayer mp) {
                        bt_play.setEnabled(true);
                    }
                } );
                
                bt_play.setEnabled(false);
            } catch (Exception e) {
                Toast.makeText(this, "播放失敗", 0).show();
                e.printStackTrace();
            }
        } else {
            Toast.makeText(this, "文件不存在", 0).show();
        }

    }

}

運行項目效果如下:

但是上面的代碼有點問題,那就是當點擊home或返回屏幕主界面後,在回到播放頁面,視頻黑屏

原因:

SurfaceView內部維護雙緩衝,消耗記憶體資源

如果發現當前SurfaceView 用戶可見的時候,創建SurfaceView的holder

如果SurfaceView變成用戶不可見的時候  銷毀SurfaceView的holder

為了觀察SurfaceView的生命周期,可以添加下麵的代碼進行列印追蹤:

    sv.getHolder().addCallback(new Callback() {
            
            @Override
            public void surfaceDestroyed(SurfaceHolder holder) {
                System.out.println("holder被銷毀了");
            }
            
            @Override
            public void surfaceCreated(SurfaceHolder holder) {
                System.out.println("holder被創建了");
                
            }
            
            @Override
            public void surfaceChanged(SurfaceHolder holder, int format, int width,
                    int height) {
                System.out.println("holder的大小變化了");
            }
        });

    }

logcat中列印下麵的信息:

09-07 11:36:34.613: I/System.out(28858): holder被創建了
09-07 11:36:34.623: I/System.out(28858): holder的大小變化了

返回主界面

09-07 11:39:08.245: I/System.out(28858): holder被銷毀了

再次回到視頻界面

09-07 11:39:39.405: I/System.out(28858): holder被創建了
09-07 11:39:39.405: I/System.out(28858): holder的大小變化了

可以看到,每次回到視頻界面,holder都會被重新創建

修改後的代碼如下:

public class MainActivity extends Activity implements OnClickListener {

    private EditText et_path;
    private Button bt_play, bt_replay, bt_pause, bt_stop;
    private SurfaceView sv;

    private MediaPlayer mediaPlayer;
    private int currentPosition;

    @Override
    protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);

        sv = (SurfaceView) findViewById(R.id.sv);
        et_path = (EditText) findViewById(R.id.et_path);
        bt_play = (Button) findViewById(R.id.bt_play);
        bt_replay = (Button) findViewById(R.id.bt_replay);
        bt_pause = (Button) findViewById(R.id.bt_pause);
        bt_stop = (Button) findViewById(R.id.bt_stop);

        bt_pause.setOnClickListener(this);
        bt_play.setOnClickListener(this);
        bt_replay.setOnClickListener(this);
        bt_stop.setOnClickListener(this);

        sv.getHolder().addCallback(new Callback() {

            @Override
            public void surfaceDestroyed(SurfaceHolder holder) {
                System.out.println("holder被銷毀了");

                if (mediaPlayer != null && mediaPlayer.isPlaying()) {
                    currentPosition = mediaPlayer.getCurrentPosition();
                    stop();
                }
            }

            @Override
            public void surfaceCreated(SurfaceHolder holder) {
                System.out.println("holder被創建了");
                if (currentPosition > 0) {
                    play(currentPosition);
                }

            }

            @Override
            public void surfaceChanged(SurfaceHolder holder, int format,
                    int width, int height) {
                System.out.println("holder的大小變化了");
            }
        });

    }

    @Override
    public void onClick(View v) {
        switch (v.getId()) {
        case R.id.bt_play:
            play(0);
            break;
        case R.id.bt_replay:
            replay();
            break;
        case R.id.bt_stop:
            stop();
            break;
        case R.id.bt_pause:
            pause();
            break;
        default:
            break;
        }
    }

    /**
     * 暫停音樂
     */
    private void pause() {
        if ("繼續".equals(bt_pause.getText().toString().trim())) {
            mediaPlayer.start();
            bt_pause.setText("暫停");
            return;
        }
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.pause();
            bt_pause.setText("繼續");
            return;
        }
    }

    /**
     * 重新播放
     */
    private void replay() {
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.seekTo(0);
            return;
        }
        play(0);
    }

    /**
     * 停止播放音樂
     */
    private void stop() {
        if (mediaPlayer != null && mediaPlayer.isPlaying()) {
            mediaPlayer.stop();
            mediaPlayer.release(); // 記得釋放資源
            mediaPlayer = null;
            bt_play.setEnabled(true);
        }
    }

    /**
     * 播放音樂
     */
    private void play(final int currentPosition) {
        String path = et_path.getText().toString().trim();
        File file = new File(path);
        if (file.exists() && file.length() > 0) {
            try {
                mediaPlayer = new MediaPlayer();
                mediaPlayer.setAudioStreamType(AudioManager.STREAM_MUSIC);

                /* 設置Video影片以SurfaceHolder播放 */
                mediaPlayer.setDisplay(sv.getHolder());

                mediaPlayer.setDataSource(path);
                mediaPlayer.prepare(); // might take long! (for buffering, etc)
                mediaPlayer.start();
                mediaPlayer.setOnPreparedListener(new OnPreparedListener() {
                    
                    @Override
                    public void onPrepared(MediaPlayer mp) {
                        mediaPlayer.seekTo(currentPosition);
                        
                    }
                });
                mediaPlayer.setOnCompletionListener(new OnCompletionListener() {
                    @Override
                    public void onCompletion(MediaPlayer mp) {
                        bt_play.setEnabled(true);
                    }
                });

                bt_play.setEnabled(false);
            } catch (Exception e) {
                Toast.makeText(this, "播放失敗", 0).show();
                e.printStackTrace();
            }
        } else {
            Toast.makeText(this, "文件不存在", 0).show();
        }

    }

}

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • Position這個屬性定義建立元素佈局所用的定位機制。任何元素都是可以進行定位的,不過絕對或者固定一個元素會產生一個塊級框,不論該元素是什麼類型;相對定位元素會相對於它在正常文檔流中的預設位置偏移。 Position元素一般來說擁有五個屬性,分別有: 1.absolute(生成絕對定位的元素,相對 ...
  • 首先Position在字面講是位置的意思,在HTML中是定位的意思,它有四種屬性:分別是static是靜態的,也是預設的效果,沒有特別的設定,遵循基本的定位規定,不能通過z-index進行層次分級。 absolute(絕對定位) absolute(絕對定位)不光脫離了文本流,而且在文本流中也不會給這 ...
  • 本文簡單介紹下如何來使用 Bootstrap,通過引入 Bootstrap,來實現一個最基本的入門例子。 在前一篇博文【Bootstrap】1.初識Bootstrap 基礎之上,我們完全可以更加方便快捷的創建一個的簡單例子。 在Bootstrap的官方網站的下載頁面 http://getbootst ...
  • 作為Web前端開發框架,Bootstrap為大多數標準的UI設計常見提供了用戶友好、擴瀏覽器的解決方案。 1.下載Bootstrap 打開官方網址 http://getbootstrap.com/ 進行下載。 2.準備項目模板文件夾 接下來,我們為第一個項目創建一個文件夾以及一些基本的文件。謂詞我們 ...
  • 語法: 作用: 啟動調試器 備註: 1. 可以將debugger語句放在過程的任何地方以中止執行。2. 使用debugger語句類似於在代碼中設置斷點。 3. debugger語句中止執行,但它不關閉任何文件或清除任何變數。 【註】只有打開調試器,否則debugger語句不起作用 實例: ...
  • × 目錄 [1]CSSStyleSheet [2]CSSRule 前面的話 關於腳本化CSS,查詢樣式時,查詢的是計算樣式;設置單個樣式時,設置的是行間樣式;設置多個樣式時,設置的是CSS類名。腳本化樣式表當然也是一種腳本化CSS的技術,雖然不經常使用,但有時卻非常有用。下麵將詳細介紹腳本化樣式表的 ...
  • 一. json介紹 json是一種輕量級的數據交換格式,規則很簡單: 1. 併列的數據之間用逗號(,)分隔; 2. 映射用冒號(:)表示; 3. 併列數據的集合(數組)用方括弧([])表示; 4. 映射的集合(對象)用大括弧({})表示。 對上述規則解析,可以發現: 數組是用([])創建的,對象是用 ...
  • Widget引入 我們可以把Widget理解成放置在桌面上的小組件(掛件),有了Widget,我們可以很方便地直接在桌面上進行各種操作,例如播放音樂。 當我們長按桌面時,可以看到Widget選項,如下圖所示: 點擊上圖中箭頭處的widgets圖標,會出現如下界面:(都是widget) 長按上圖中的任 ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...