一起學Android之音頻視頻

来源:https://www.cnblogs.com/hsiang/archive/2019/07/13/11181673.html
-Advertisement-
Play Games

本文演示瞭如何通過Api播放音頻和視頻文件,來獲得良好性能和愉快的用戶體驗,僅供學習分享使用,如有不足之處,還請指正。 ...


概述

Android多媒體框架支持各種常見的媒體類型,可以很容易地將音頻、視頻和圖像集成到App中。通過MediaPlayer Api,可以從應用程式資源(RAW)、文件系統或網路上數據流資源來播放音頻或視頻。本文演示瞭如何通過Api播放音頻和視頻文件,來獲得良好性能和愉快的用戶體驗,僅供學習分享使用,如有不足之處,還請指正。

涉及知識點

  1.  MediaPlayer 可以用來控制audio/video文件或流播放的類。通過此類,可以方便的控制音頻/視頻文件的播放,暫停和停止等操作。
  2. Uri 統一資源標識符(Uniform Resource Identifier,或URI)是一個用於標識某一互聯網資源名稱的字元串。
  3. VideoView 自帶的一種播放視頻的組件。
  4. SurfaceView 可以通過後臺繪製顯示的視圖組件。 

Activity中播放音頻

 頁面上播放音頻文件,步驟如下:

1. 準備資源文件,並播放

 1 private MediaPlayer mMediaPlayer;
 2 
 3     /**
 4      * 開始
 5      * @param v
 6      */
 7     public void bn_start(View v){
 8         if(mMediaPlayer==null) {
 9             ready();
10         }
11         mMediaPlayer.start();
12     }
13 
14  /**
15      * 創建MediaPlayer併到prepare狀態
16      */
17     private void ready(){
18         if(mMediaPlayer==null){
19             mMediaPlayer=new MediaPlayer();//Idle
20             //Log.i("TAG", "ready: "+filePath);
21             File file=new File(Environment.getExternalStorageDirectory(),"goldfallen.mp3");
22             Log.i("TAG", "ready: "+file.getPath());
23             Log.i("TAG", "ready: "+file.exists());
24             Uri uri= Uri.fromFile(file);
25             try {
26                 mMediaPlayer.setDataSource(AudioActivity.this,uri);
27                 mMediaPlayer.prepare();//Prepared
28             } catch (IOException e) {
29                 Log.i("TAG", "ready: "+ e.getMessage());
30                 e.printStackTrace();
31             }
32         }
33     }

2. 暫停播放

1 /**
2      * 暫停
3      * @param v
4      */
5     public void bn_pause(View v){
6         if(mMediaPlayer!=null && mMediaPlayer.isPlaying()){
7             mMediaPlayer.pause();
8         }
9     }

3. 停止

 1   /**
 2      * 停止
 3      * @param v
 4      */
 5     public void bn_stop(View v){
 6         if(mMediaPlayer!=null && mMediaPlayer.isPlaying()){
 7             mMediaPlayer.stop();
 8             mMediaPlayer.release();
 9             mMediaPlayer=null;
10         }
11     }

4. 釋放資源(當頁面銷毀時,播放資源也要同時釋放)

1  @Override
2     protected void onDestroy() {
3         if(mMediaPlayer!=null && mMediaPlayer.isPlaying()){
4             mMediaPlayer.stop();
5             mMediaPlayer.release();
6             mMediaPlayer=null;
7         }
8         super.onDestroy();
9     }

5. 通過靜態函數創建對象

 1 /**
 2      * D:\Android\Project\DemoMedia\app\src\main\res\raw\A2012.mp3:
 3      * Error: 'A' is not a valid file-based resource name character:
 4      * File-based resource names must contain only lowercase a-z, 0-9, or underscore
 5      */
 6     private void ready2(){
 7         if(mMediaPlayer==null){
 8             mMediaPlayer= MediaPlayer.create(this,R.raw.ab2012);
 9         }
10     }

通過服務來播放音頻

當頁面關閉時,音頻文件還可以在後臺播放。步驟如下:

1. 定義後臺服務,當服務啟動時,創建MediaPlayer對象,並註冊服務接收對象。

 1 public class AudioService extends Service {
 2 
 3     private MediaPlayer mMediaPlayer;
 4 
 5     private OperatorReceiver mReceiver;
 6 
 7     public AudioService() {
 8     }
 9 
10     @Override
11     public void onCreate() {
12         super.onCreate();
13         mMediaPlayer=new MediaPlayer();
14         mReceiver=new OperatorReceiver();
15         IntentFilter filter=new IntentFilter("operator.receiver");
16         registerReceiver(mReceiver,filter);
17     }
18 
19     @Override
20     public IBinder onBind(Intent intent) {
21         return null;
22     }
23 
24     @Override
25     public int onStartCommand(Intent intent, int flags, int startId) {
26         Log.i("TAG", "onStartCommand: ----ready ");
27         ready();
28         return super.onStartCommand(intent, flags, startId);
29     }
30 
31     @Override
32     public void onDestroy() {
33         if(mMediaPlayer!=null && mMediaPlayer.isPlaying()){
34             mMediaPlayer.stop();
35         }
36         if(mMediaPlayer!=null){
37             mMediaPlayer.release();
38             mMediaPlayer=null;
39         }
40         unregisterReceiver(mReceiver);
41         super.onDestroy();
42     }
43 
44     private void start(){
45         mMediaPlayer.start();
46     }
47 
48     private  void pause(){
49         mMediaPlayer.pause();
50     }
51 
52     private void stop(){
53         mMediaPlayer.stop();
54         ready();
55     }
56 
57     private void ready(){
58         mMediaPlayer.reset();
59         try {
60             String filePath= Environment.getExternalStorageDirectory()+"/ab2012.mp3";
61             Log.i("TAG", "ready: "+filePath);
62             File file=new File(filePath);
63             if(file.exists()) {
64                 mMediaPlayer.setDataSource(filePath);
65                 mMediaPlayer.prepare();
66             }else{
67                 Log.i("TAG", "ready: 文件不存在 ");
68             }
69         } catch (IOException e) {
70             Log.i("TAG", "ready: "+e.getMessage());
71             e.printStackTrace();
72         }
73     }
74 }

2. 操作接收者定義如下:

 1  /**
 2      * 操作接收器
 3      */
 4     class OperatorReceiver extends BroadcastReceiver
 5     {
 6         @Override
 7         public void onReceive(Context context, Intent intent) {
 8             int cmd=intent.getIntExtra("cmd",-1);
 9             Log.i("TAG", "onReceive: "+cmd);
10             switch (cmd){
11                 case Tools.PLAY:
12                     start();
13                     break;
14                 case Tools.PAUSE:
15                     pause();
16                     break;
17                 case Tools.STOP:
18                     stop();
19                     break;
20                 default:
21                     start();
22                     break;
23 
24             }
25         }
26     }

3. 啟動服務

1   Intent intent =new Intent(this,AudioService.class);
2   startService(intent);

4. 前臺Activity通過發送廣播命令進行控制音頻的播放與暫停

 1 /**
 2      * 開始
 3      * @param v
 4      */
 5     public void bn_start(View v){
 6         Intent intent=new Intent("operator.receiver");
 7         intent.putExtra("cmd",Tools.PLAY);
 8         sendBroadcast(intent);
 9         Log.i("TAG", "bn_start: "+"service");
10     }
11 
12     /**
13      * 暫停
14      * @param v
15      */
16     public void bn_pause(View v){
17         Intent intent=new Intent("operator.receiver");
18         intent.putExtra("cmd",Tools.PAUSE);
19         sendBroadcast(intent);
20     }
21 
22     /**
23      * 停止
24      * @param v
25      */
26     public void bn_stop(View v){
27         Intent intent=new Intent("operator.receiver");
28         intent.putExtra("cmd",Tools.STOP);
29         sendBroadcast(intent);
30     }

5.通過後臺服務的方式,需要在AndroidManifest.xml文件中進行註冊

1  <service
2             android:name=".AudioService"
3             android:enabled="true"
4             android:exported="true">
5         </service>

通過VideoView來播放視頻

 通過VideoView進行播放視頻文件,只需要設置資源URI即可,具體如下:

 1  private VideoView mVideoView;
 2     @Override
 3     protected void onCreate(Bundle savedInstanceState) {
 4         super.onCreate(savedInstanceState);
 5         setContentView(R.layout.activity_video);
 6         mVideoView= (VideoView) this.findViewById(R.id.vv_info);
 7         //設置路徑
 8         File file=new File(Environment.getExternalStorageDirectory(),"DCIM/Camera/VID20190629142614.mp4");
 9         Log.i("TAG", "onCreate: "+file.getPath());
10         if(!file.exists()){
11             Log.i("TAG", "onCreate: 文件不存在");
12         }
13         Uri uri=Uri.fromFile(file);
14         mVideoView.setVideoURI(uri);
15         //設置Media Controller
16         MediaController controller=new MediaController(this);
17         mVideoView.setMediaController(controller);
18         //獲取焦點
19         mVideoView.requestFocus();
20         mVideoView.start();
21         //設置播放完成事件
22         mVideoView.setOnCompletionListener(new MediaPlayer.OnCompletionListener() {
23             @Override
24             public void onCompletion(MediaPlayer mp) {
25                 Toast.makeText(VideoActivity.this, "播放完畢", Toast.LENGTH_SHORT).show();
26             }
27         });
28     }

通過MediaPlayer來播放視頻

通過MediaPlayer來播放視頻文件,然後通過SurfaceView來顯示視頻內容,具體步驟如下:

1.啟動,通過setDisplay來設置顯示內容

 1 public  void bn_start(View view)  {
 2         mMediaPlayer.reset();//到Idle狀態
 3         //設置路徑
 4         File file=new File(Environment.getExternalStorageDirectory(),"DCIM/Camera/VID20190629142614.mp4");
 5         Log.i("TAG", "onCreate: "+file.getPath());
 6         if(!file.exists()){
 7             Log.i("TAG", "onCreate: 文件不存在");
 8         } else {
 9             Log.i("TAG", "onCreate: 文件存在 ");
10         }
11         Uri uri=Uri.fromFile(file);
12         try {
13             mMediaPlayer.setDataSource(this,uri);
14             mMediaPlayer.setDisplay(mSurfaceView.getHolder());
15             mMediaPlayer.prepare();
16             mMediaPlayer.start();
17         } catch (IOException e) {
18             e.printStackTrace();
19         }
20     }

2. 暫停與停止

 1  public  void bn_pause(View view){
 2         if(mMediaPlayer.isPlaying()){
 3             mMediaPlayer.pause();
 4         }
 5     }
 6 
 7     public  void bn_stop(View view){
 8         if(mMediaPlayer.isPlaying()){
 9             mMediaPlayer.stop();
10         }
11     }

3. 資源釋放

 1  @Override
 2     protected void onDestroy() {
 3         if(mMediaPlayer!=null && mMediaPlayer.isPlaying()){
 4             mMediaPlayer.stop();
 5         }
 6         if(mMediaPlayer!=null){
 7             mMediaPlayer.release();
 8             mMediaPlayer=null;
 9         }
10         super.onDestroy();
11     }

許可權設置

如果要播放視頻,需要相應的許可權設置,併在安裝時獲取請求許可權

1     <uses-permission android:name="android.permission.INTERNET"/>
2     <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>
3     <uses-permission android:name="android.permission.READ_EXTERNAL_STORAGE"/>
4     <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"/>

備註

合抱之木,生於毫末;九層之台,起於壘土;千里之行,始於足下。

 


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

-Advertisement-
Play Games
更多相關文章
  • 最近在使用遇到一個問題需要把csv格式的文件轉成xls,隨便新建一個excel,然後打開,選擇“開發工具”,找到下圖“巨集”,如果跟下圖一樣的話就需要先啟用巨集,啟用之後可以直接把下麵的代碼直接複製到代碼區,修改一個路徑就好。 以下代碼試講文件名為1.csv的文件,轉化為2.xls ok,今天的分享就到 ...
  • 前言:源碼預編譯MySQL資料庫,使用時cmake 方式,MySQL資料庫官方出的資料庫編譯命令,和普通源碼安裝軟體不同 (configure)。 CMake是一個跨平臺的安裝(編譯)工具,可以用簡單的語句來描述所有平臺的安裝(編譯過程)。cmake所做的事其實就 是告訴編譯器如何去編譯鏈接源代碼。 ...
  • Redis的List數據類型作為消息隊列,已經比較合適了,但存在一些不足,比如只能獨立消費,訂閱發佈又無法支持數據的持久化,相對前兩者,Redis Stream作為消息隊列的使用更為有優勢。 相信球迷小伙伴們對文字直播這個東西都不陌生,時常在想,這個功能是怎麼實現的? 具體說就是用什麼技術實現最為合 ...
  • 單表查詢 [TOC] 準備數據 只是把查詢結果按照自己想要的方式返回,不對數據做修改 簡單查詢 指定欄位查詢 select id ,emp_name,sex,age,hire_date,post,post_comment,salary,office,depart_id from employee; ...
  • mysql 資料庫 規範 [TOC] 基礎規範 必須使用InnoDB存儲引擎 解讀:支持事務、行級鎖、併發性能更好、CPU及記憶體緩存頁優化使得資源利用率更高 新庫使用utf8mb4字元集 解讀:萬國碼,無需轉碼,無亂碼風險,節省空間 數據表、數據欄位必須加入中文註釋 解讀:N年後誰tm知道這個r1, ...
  • 本人在寫一個測試demo的時候,遇到一個問題就是添加的中文數據在資料庫定義的明明是varchar類型,但是顯示出來還是亂碼,不過還是自己粗心導致的問題。 以下三種方式可以自查一下: 1、 首先檢查資料庫,字元集的類型是否是utf8_general_ci類型; 2、連接字元串,在連接字元串最後面加上 ...
  • # 檢查Java版本 java -version # 安裝Elasticsearch,所有節點均安裝並解壓 wget https://artifacts.elastic.co/downloads/elasticsearch/elasticsearch6.3.0.tar.gz 2.tar -xf el ...
  • 針對手機APP開發人員來說,很多時候開發手機APP應用的時候,需要進入手機的開發者模式中對程式進行調試等操作,此文將介紹華為榮耀手機如何進入開發者模式,以華為榮耀V9手機為例,華為榮耀V10、榮耀V20、華為P20、華為P30等型號可以參考此教程,具體以官方的教程為準。 (1)首先進入手機的設置項“ ...
一周排行
    -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.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...