Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> Android開發實例 >> Android音樂播放器的開發實例

Android音樂播放器的開發實例

編輯:Android開發實例

       本文將引導大家做一個音樂播放器,在做這個Android開發實例的過程中,能夠幫助大家進一步熟悉和掌握學過的ListView和其他一些組件。為了有更好的學習效果,其中很多功能我們手動實現,例如音樂播放的快進快退等。

       先欣賞下本實例完成後運行的界面效果:

Android音樂播放器的開發實例

        首先我們建立項目,我使用的SDK是Android2.2的,然後在XML中進行布局。

       上方是一個ListView用來顯示我們的音樂列表,中間是一個SeekBar可以拖動當前音樂的播放進度,之所以用SeekBar而不用ProgressBar是因為我們需要音樂的快進快退功能,可以拖動滑桿改變進度;還有一個TextView,用來顯示當前播放歌曲的名字,時長等。最下方就是4個Button了,分別是上一曲,播放(暫停),停止,下一曲。

       大家注意盡量不要在布局中出現直接顯示在界面上的文字內容,我們把這些內容都放在res/values下的strings.xml中,然後分別引用它們,這樣養成良好的習慣,界面與內容分離,方便調試和後期維護等。現在我們的界面如下:

Android音樂播放器XML布局

       然後我們把File Explorer打開,在eclipse的Window -- Show View -- Other --Android --File Explore。你也可以直接Alt+Shift+Q。

File Explorer

       在mnt/sdcard下面,我們放個兩三首歌曲,在虛擬機中暫不支持中文,導入有中文的文件會報錯的。

       接著我們創建一個類,做我們播放器的Service類,我就叫MusicService吧,在裡面聲明以下對象:

Java代碼
  1. public class MusicService {   
  2.   
  3.     private static final File MUSIC_PATH = Environment   
  4.             .getExternalStorageDirectory();// 找到music存放的路徑。   
  5.     public List<String> musicList;// 存放找到的所有mp3的絕對路徑。   
  6.     public MediaPlayer player; // 定義多媒體對象   
  7.     public int songNum; // 當前播放的歌曲在List中的下標   
  8.     public String songName; // 當前播放的歌曲名   
  9.   
  10. }  

       然後我們去加載剛才添加的MP3文件吧,這裡的方式多種多樣,我隨便寫一個簡單的了:

Java代碼
  1. class MusicFilter implements FilenameFilter {   
  2.      public boolean accept(File dir, String name) {   
  3.      return (name.endsWith(".mp3"));//返回當前目錄所有以.mp3結尾的文件   
  4.      }   
  5. }  

       在MusicService類的無參構造函數中實例化對象,並把這些MP3文件放到musicList中。

Java代碼
  1. public MusicService() {   
  2.     musicList = new ArrayList<String>();   
  3.     player = new MediaPlayer();   
  4.   
  5.     if (MUSIC_PATH.listFiles(new MusicFilter()).length > 0) {   
  6.         for (File file : MUSIC_PATH.listFiles(new MusicFilter())) {   
  7.             musicList.add(file.getAbsolutePath());   
  8.         }   
  9.     }   
  10. }  

       我們寫個方法,來設置當前播放歌曲的名字:(個人覺得這方法比較笨,但暫時沒想到別的辦法)

Java代碼
  1. public void setPlayName(String dataSource) {   
  2.     File file = new File(dataSource);//假設為D:\\mm.mp3   
  3.     String name = file.getName();//name=mm.mp3   
  4.     int index = name.lastIndexOf(".");//找到最後一個.   
  5.     songName = name.substring(0, index);//截取為mm   
  6. }  

      接下來就是我們Service類的基本方法了,也就是開始、暫停、停止、上一首和下一首。

      我們分別使用聲明的多媒體對象的start、pause、stop等方法可以完成。

Java代碼
  1. public void start() {   
  2.     try {   
  3.         player.reset(); //重置多媒體   
  4.         String dataSource = musicList.get(songNum);//得到當前播放音樂的路徑   
  5.         setPlayName(dataSource);//截取歌名   
  6.         player.setDataSource(dataSource);//為多媒體對象設置播放路徑   
  7.         player.prepare();//准備播放   
  8.         player.start();//開始播放   
  9.         //setOnCompletionListener 當當前多媒體對象播放完成時發生的事件   
  10.         player.setOnCompletionListener(new OnCompletionListener() {   
  11.             public void onCompletion(MediaPlayer arg0) {   
  12.                 next();//如果當前歌曲播放完畢,自動播放下一首.   
  13.             }   
  14.         });   
  15.     } catch (Exception e) {   
  16.         Log.v("MusicService", e.getMessage());   
  17.     }   
  18. }   
  19.   
  20. public void next() {   
  21.     songNum = songNum == musicList.size() - 1 ? 0 : songNum + 1;   
  22.     start();   
  23. }   
  24.   
  25. public void last() {   
  26.     songNum = songNum == 0 ? musicList.size() - 1 : songNum - 1;   
  27.     start();   
  28. }   
  29.   
  30. public void pause() {   
  31.     if (player.isPlaying())   
  32.         player.pause();   
  33.     else  
  34.         player.start();   
  35. }   
  36.   
  37. public void stop() {   
  38.     if (player.isPlaying()) {   
  39.         player.stop();   
  40.     }   
  41. }  

       到此為止我們的Service類就寫完了,接著我們去Activity中為各控件綁定事件。

       在這個Activity中,最難做的一點應該就是拖動SeekBar的滑桿改變播放進度了,這裡我考慮再三,用了一個Handler類來處理。

       Handler在android裡負責發送和處理消息。它的主要用途有:

       1.按計劃發送消息或執行某個Runnanble(使用POST方法)。

       2.從其他線程中發送來的消息放入消息隊列中,避免線程沖突(常見於更新UI線程)。

       默認情況下,Handler接受的是當前線程下的消息循環實例(使用Handler(Looper looper)、Handler(Looper looper, Handler.Callback callback)可以指定線程),同時一個消息隊列可以被當前線程中的多個對象進行分發、處理(在UI線程中,系統已經有一個Activity來處理了,你可以再起若干個Handler來處理)。在實例化Handler的時候,Looper可以是任意線程的,只要有Handler的指針,任何線程也都可以sendMessage。Handler對於Message的處理不是並發的。一個Looper 只有處理完一條Message才會讀取下一條,所以消息的處理是阻塞形式的(handleMessage()方法裡不應該有耗時操作,可以將耗時操作放在其他線程執行,操作完後發送Message(通過sendMessges方法),然後由handleMessage()更新UI)。

       聲明以下變量:

Java代碼
  1. private Button btnStart, btnStop, btnNext, btnLast;   
  2. private TextView txtInfo;   
  3. private ListView listView;   
  4. private SeekBar seekBar;   
  5. private MusicService musicService;   
  6. private MusicHandler musicHandler;// 處理改變進度條事件   
  7. private MusicThread musicThread;// 自動改變進度條的線程   
  8. private boolean autoChange, manulChange;// 判斷是進度條是自動改變還是手動改變   
  9. private boolean isPause;// 判斷是從暫停中恢復還是重新播放  

       如有報錯的可以先注釋掉不用管它,然後在初始化過程中綁定事件。

       這是ListView的填充方法:

Java代碼
  1. private void setListViewAdapter() {   
  2.     List<Map<String, Object>> date = new ArrayList<Map<String, Object>>();   
  3.   
  4.     for (String path : musicService.musicList) {   
  5.         Map<String, Object> map = new HashMap<String, Object>();   
  6.         File file = new File(path);   
  7.         map.put("fileName", file.getName());   
  8.         date.add(map);   
  9.     }   
  10.     SimpleAdapter adapter = new SimpleAdapter(this, date,   
  11.                 android.R.layout.simple_list_item_1,   
  12.                 new String[] { "fileName" }, new int[] { android.R.id.text1 });   
  13.   
  14.     listView.setAdapter(adapter);   
  15.   
  16. }  

       SimpleAdapter的構造函數是:

       public SimpleAdapter (Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to);

       第一個參數context,是指在哪個Activity中顯示。

       第二個參數是一個泛型作為數據源,而且每一個List中的一行就代表著呈現出來的一行,Map的鍵就是這一行的列名,值也是有列名的。

       第三個參數為資源文件,就是說要加載這個列所需要的視圖資源文件,我直接引用系統內置的資源,如果你想要漂亮的樣式可以自己寫的。

       第四個參數是一個String數組,主要是將Map對象中的名稱映射到列名,一一對應。

       第五個是將第四個參數的值一一對象的顯示(一一對應)在接下來的int形的id數組中,這個id數組就是Layout的xml文件中命名id形成的唯一的int型標識符。

       SeekBar停止拖動後的事件:

Java代碼
  1. public void onStopTrackingTouch(SeekBar seekBar) { // 停止拖動      
  2.     int progress = seekBar.getProgress();      
  3.      
  4.     if (!autoChange && manulChange) {      
  5.         int musicMax = musicService.player.getDuration(); //得到該首歌曲最長秒數      
  6.         int seekBarMax = seekBar.getMax();      
  7.      
  8.         musicService.player      
  9.                 .seekTo(musicMax * progress / seekBarMax);//跳到該曲該秒                     
  10.     musicService.pause();      
  11.     autoChange = true;      
  12.     manulChange = false;      
  13.     }      
  14. }    

       MusicHandler類的實現:

Java代碼
  1. class MusicHandler extends Handler {   
  2.   
  3.          public MusicHandler() {   
  4.     }   
  5.   
  6.     @Override  
  7.     public void handleMessage(Message msg) {   
  8.         if (autoChange) {   
  9.             try {   
  10.                 int position = musicService.player.getCurrentPosition();//得到當前歌曲播放進度(秒)   
  11.                 int mMax = musicService.player.getDuration();//最大秒數   
  12.                 int sMax = seekBar.getMax();//seekBar最大值,算百分比   
  13.                     seekBar.setProgress(position * sMax / mMax);   
  14.                     txtInfo.setText(setPlayInfo(position / 1000, mMax / 1000));   
  15.             } catch (Exception e) {   
  16.                     e.printStackTrace();   
  17.             }   
  18.         } else {   
  19.             seekBar.setProgress(0);   
  20.             txtInfo.setText("播放已經停止");   
  21.         }   
  22.     }   
  23. }   
  24.   
  25. //設置當前播放的信息   
  26. private String setPlayInfo(int position, int max) {   
  27.     String info = "正在播放:  " + musicService.songName + "\t\t";   
  28.   
  29.     //笨辦法 寫完才想起可以用%的,但不想改了   
  30.     int pMinutes = 0;   
  31.     while (position >= 60) {   
  32.         pMinutes++;   
  33.         position -= 60;   
  34.     }   
  35.     String now = (pMinutes < 10 ? "0" + pMinutes : pMinutes) + ":"  
  36.         + (position < 10 ? "0" + position : position);   
  37.   
  38.     int mMinutes = 0;   
  39.     while (max >= 60) {   
  40.         mMinutes++;   
  41.         max -= 60;   
  42.     }   
  43.     String all = (mMinutes < 10 ? "0" + mMinutes : mMinutes) + ":"  
  44.         + (max < 10 ? "0" + max : max);   
  45.   
  46.     return info + now + " / " + all;   
  47. }  

       MusicThread的實現:

Java代碼
  1. class MusicThread implements Runnable {   
  2.   
  3.     @Override  
  4.     public void run() {   
  5.         while (true)   
  6.             try {   
  7.                     musicHandler.sendMessage(new Message());   
  8.                 Thread.sleep(1000);// 每間隔1秒發送一次更新消息   
  9.             } catch (InterruptedException e) {   
  10.                     e.printStackTrace();   
  11.             }   
  12.     }   
  13.   
  14. }  

       至此項目完成。希望大家能從這個實例中學到更多的東西,積累更多經驗。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved