Android 音樂(lè)播放器的開(kāi)發(fā)實(shí)例詳解
本文將引導(dǎo)大家做一個(gè)音樂(lè)播放器,在做這個(gè)Android開(kāi)發(fā)實(shí)例的過(guò)程中,能夠幫助大家進(jìn)一步熟悉和掌握學(xué)過(guò)的ListView和其他一些組件。為了有更好的學(xué)習(xí)效果,其中很多功能我們手動(dòng)實(shí)現(xiàn),例如音樂(lè)播放的快進(jìn)快退等。
先欣賞下本實(shí)例完成后運(yùn)行的界面效果:
首先我們建立項(xiàng)目,我使用的SDK是Android2.2的,然后在XML中進(jìn)行布局。
上方是一個(gè)ListView用來(lái)顯示我們的音樂(lè)列表,中間是一個(gè)SeekBar可以拖動(dòng)當(dāng)前音樂(lè)的播放進(jìn)度,之所以用SeekBar而不用ProgressBar是因?yàn)槲覀冃枰魳?lè)的快進(jìn)快退功能,可以拖動(dòng)滑桿改變進(jìn)度;還有一個(gè)TextView,用來(lái)顯示當(dāng)前播放歌曲的名字,時(shí)長(zhǎng)等。最下方就是4個(gè)Button了,分別是上一曲,播放(暫停),停止,下一曲。
大家注意盡量不要在布局中出現(xiàn)直接顯示在界面上的文字內(nèi)容,我們把這些內(nèi)容都放在res/values下的strings.xml中,然后分別引用它們,這樣養(yǎng)成良好的習(xí)慣,界面與內(nèi)容分離,方便調(diào)試和后期維護(hù)等?,F(xiàn)在我們的界面如下:
然后我們把File Explorer打開(kāi),在eclipse的Window -- Show View -- Other --Android --File Explore。你也可以直接Alt+Shift+Q。
在mnt/sdcard下面,我們放個(gè)兩三首歌曲,在虛擬機(jī)中暫不支持中文,導(dǎo)入有中文的文件會(huì)報(bào)錯(cuò)的。
接著我們創(chuàng)建一個(gè)類,做我們播放器的Service類,我就叫MusicService吧,在里面聲明以下對(duì)象:
Java代碼
public class MusicService { private static final File MUSIC_PATH = Environment .getExternalStorageDirectory();// 找到music存放的路徑。 public List<String> musicList;// 存放找到的所有mp3的絕對(duì)路徑。 public MediaPlayer player; // 定義多媒體對(duì)象 public int songNum; // 當(dāng)前播放的歌曲在List中的下標(biāo) public String songName; // 當(dāng)前播放的歌曲名 }
然后我們?nèi)ゼ虞d剛才添加的MP3文件吧,這里的方式多種多樣,我隨便寫(xiě)一個(gè)簡(jiǎn)單的了:
Java代碼
class MusicFilter implements FilenameFilter { public boolean accept(File dir, String name) { return (name.endsWith(".mp3"));//返回當(dāng)前目錄所有以.mp3結(jié)尾的文件 } }
在MusicService類的無(wú)參構(gòu)造函數(shù)中實(shí)例化對(duì)象,并把這些MP3文件放到musicList中。
Java代碼
public MusicService() { musicList = new ArrayList<String>(); player = new MediaPlayer(); if (MUSIC_PATH.listFiles(new MusicFilter()).length > 0) { for (File file : MUSIC_PATH.listFiles(new MusicFilter())) { musicList.add(file.getAbsolutePath()); } } }
我們寫(xiě)個(gè)方法,來(lái)設(shè)置當(dāng)前播放歌曲的名字:(個(gè)人覺(jué)得這方法比較笨,但暫時(shí)沒(méi)想到別的辦法)
Java代碼
public void setPlayName(String dataSource) { File file = new File(dataSource);//假設(shè)為D:\\mm.mp3 String name = file.getName();//name=mm.mp3 int index = name.lastIndexOf(".");//找到最后一個(gè). songName = name.substring(0, index);//截取為mm }
接下來(lái)就是我們Service類的基本方法了,也就是開(kāi)始、暫停、停止、上一首和下一首。
我們分別使用聲明的多媒體對(duì)象的start、pause、stop等方法可以完成。
Java代碼
public void start() { try { player.reset(); //重置多媒體 String dataSource = musicList.get(songNum);//得到當(dāng)前播放音樂(lè)的路徑 setPlayName(dataSource);//截取歌名 player.setDataSource(dataSource);//為多媒體對(duì)象設(shè)置播放路徑 player.prepare();//準(zhǔn)備播放 player.start();//開(kāi)始播放 //setOnCompletionListener 當(dāng)當(dāng)前多媒體對(duì)象播放完成時(shí)發(fā)生的事件 player.setOnCompletionListener(new OnCompletionListener() { public void onCompletion(MediaPlayer arg0) { next();//如果當(dāng)前歌曲播放完畢,自動(dòng)播放下一首. } }); } catch (Exception e) { Log.v("MusicService", e.getMessage()); } } public void next() { songNum = songNum == musicList.size() - 1 ? 0 : songNum + 1; start(); } public void last() { songNum = songNum == 0 ? musicList.size() - 1 : songNum - 1; start(); } public void pause() { if (player.isPlaying()) player.pause(); else player.start(); } public void stop() { if (player.isPlaying()) { player.stop(); } }
到此為止我們的Service類就寫(xiě)完了,接著我們?nèi)ctivity中為各控件綁定事件。
在這個(gè)Activity中,最難做的一點(diǎn)應(yīng)該就是拖動(dòng)SeekBar的滑桿改變播放進(jìn)度了,這里我考慮再三,用了一個(gè)Handler類來(lái)處理。
Handler在android里負(fù)責(zé)發(fā)送和處理消息。它的主要用途有:
1.按計(jì)劃發(fā)送消息或執(zhí)行某個(gè)Runnanble(使用POST方法)。
2.從其他線程中發(fā)送來(lái)的消息放入消息隊(duì)列中,避免線程沖突(常見(jiàn)于更新UI線程)。
默認(rèn)情況下,Handler接受的是當(dāng)前線程下的消息循環(huán)實(shí)例(使用Handler(Looper looper)、Handler(Looper looper, Handler.Callback callback)可以指定線程),同時(shí)一個(gè)消息隊(duì)列可以被當(dāng)前線程中的多個(gè)對(duì)象進(jìn)行分發(fā)、處理(在UI線程中,系統(tǒng)已經(jīng)有一個(gè)Activity來(lái)處理了,你可以再起若干個(gè)Handler來(lái)處理)。在實(shí)例化Handler的時(shí)候,Looper可以是任意線程的,只要有Handler的指針,任何線程也都可以sendMessage。Handler對(duì)于Message的處理不是并發(fā)的。一個(gè)Looper 只有處理完一條Message才會(huì)讀取下一條,所以消息的處理是阻塞形式的(handleMessage()方法里不應(yīng)該有耗時(shí)操作,可以將耗時(shí)操作放在其他線程執(zhí)行,操作完后發(fā)送Message(通過(guò)sendMessges方法),然后由handleMessage()更新UI)。
聲明以下變量:
Java代碼
private Button btnStart, btnStop, btnNext, btnLast; private TextView txtInfo; private ListView listView; private SeekBar seekBar; private MusicService musicService; private MusicHandler musicHandler;// 處理改變進(jìn)度條事件 private MusicThread musicThread;// 自動(dòng)改變進(jìn)度條的線程 private boolean autoChange, manulChange;// 判斷是進(jìn)度條是自動(dòng)改變還是手動(dòng)改變 private boolean isPause;// 判斷是從暫停中恢復(fù)還是重新播放
如有報(bào)錯(cuò)的可以先注釋掉不用管它,然后在初始化過(guò)程中綁定事件。
這是ListView的填充方法:
Java代碼
private void setListViewAdapter() { List<Map<String, Object>> date = new ArrayList<Map<String, Object>>(); for (String path : musicService.musicList) { Map<String, Object> map = new HashMap<String, Object>(); File file = new File(path); map.put("fileName", file.getName()); date.add(map); } SimpleAdapter adapter = new SimpleAdapter(this, date, android.R.layout.simple_list_item_1, new String[] { "fileName" }, new int[] { android.R.id.text1 }); listView.setAdapter(adapter); }
SimpleAdapter的構(gòu)造函數(shù)是:
public SimpleAdapter (Context context, List<? extends Map<String, ?>> data, int resource, String[] from, int[] to);
第一個(gè)參數(shù)context,是指在哪個(gè)Activity中顯示。
第二個(gè)參數(shù)是一個(gè)泛型作為數(shù)據(jù)源,而且每一個(gè)List中的一行就代表著呈現(xiàn)出來(lái)的一行,Map的鍵就是這一行的列名,值也是有列名的。
第三個(gè)參數(shù)為資源文件,就是說(shuō)要加載這個(gè)列所需要的視圖資源文件,我直接引用系統(tǒng)內(nèi)置的資源,如果你想要漂亮的樣式可以自己寫(xiě)的。
第四個(gè)參數(shù)是一個(gè)String數(shù)組,主要是將Map對(duì)象中的名稱映射到列名,一一對(duì)應(yīng)。
第五個(gè)是將第四個(gè)參數(shù)的值一一對(duì)象的顯示(一一對(duì)應(yīng))在接下來(lái)的int形的id數(shù)組中,這個(gè)id數(shù)組就是Layout的xml文件中命名id形成的唯一的int型標(biāo)識(shí)符。
SeekBar停止拖動(dòng)后的事件:
Java代碼
public void onStopTrackingTouch(SeekBar seekBar) { // 停止拖動(dòng) int progress = seekBar.getProgress(); if (!autoChange && manulChange) { int musicMax = musicService.player.getDuration(); //得到該首歌曲最長(zhǎng)秒數(shù) int seekBarMax = seekBar.getMax(); musicService.player .seekTo(musicMax * progress / seekBarMax);//跳到該曲該秒 musicService.pause(); autoChange = true; manulChange = false; } }
MusicHandler類的實(shí)現(xiàn):
Java代碼
class MusicHandler extends Handler { public MusicHandler() { } @Override public void handleMessage(Message msg) { if (autoChange) { try { int position = musicService.player.getCurrentPosition();//得到當(dāng)前歌曲播放進(jìn)度(秒) int mMax = musicService.player.getDuration();//最大秒數(shù) int sMax = seekBar.getMax();//seekBar最大值,算百分比 seekBar.setProgress(position * sMax / mMax); txtInfo.setText(setPlayInfo(position / 1000, mMax / 1000)); } catch (Exception e) { e.printStackTrace(); } } else { seekBar.setProgress(0); txtInfo.setText("播放已經(jīng)停止"); } } } //設(shè)置當(dāng)前播放的信息 private String setPlayInfo(int position, int max) { String info = "正在播放: " + musicService.songName + "\t\t"; //笨辦法 寫(xiě)完才想起可以用%的,但不想改了 int pMinutes = 0; while (position >= 60) { pMinutes++; position -= 60; } String now = (pMinutes < 10 ? "0" + pMinutes : pMinutes) + ":" + (position < 10 ? "0" + position : position); int mMinutes = 0; while (max >= 60) { mMinutes++; max -= 60; } String all = (mMinutes < 10 ? "0" + mMinutes : mMinutes) + ":" + (max < 10 ? "0" + max : max); return info + now + " / " + all; }
MusicThread的實(shí)現(xiàn):
Java代碼
class MusicThread implements Runnable { @Override public void run() { while (true) try { musicHandler.sendMessage(new Message()); Thread.sleep(1000);// 每間隔1秒發(fā)送一次更新消息 } catch (InterruptedException e) { e.printStackTrace(); } } }
至此項(xiàng)目完成。希望大家能從這個(gè)實(shí)例中學(xué)到更多的東西,積累更多經(jīng)驗(yàn)。
以上就是關(guān)于Android 開(kāi)發(fā)簡(jiǎn)單的播放器實(shí)例,謝謝大家對(duì)本站的支持!
- Android實(shí)現(xiàn)簡(jiǎn)單音樂(lè)播放器(MediaPlayer)
- Android簡(jiǎn)易音樂(lè)播放器實(shí)現(xiàn)代碼
- 教你輕松制作Android音樂(lè)播放器
- android暫?;蛲V蛊渌魳?lè)播放器的播放實(shí)現(xiàn)代碼
- Android編程開(kāi)發(fā)音樂(lè)播放器實(shí)例
- Android音樂(lè)播放器制作 掃描本地音樂(lè)顯示在手機(jī)(一)
- android實(shí)現(xiàn)音樂(lè)播放器進(jìn)度條效果
- Android MediaPlayer實(shí)現(xiàn)音樂(lè)播放器實(shí)例代碼
- 簡(jiǎn)單實(shí)現(xiàn)Android本地音樂(lè)播放器
- Android實(shí)現(xiàn)簡(jiǎn)單的音樂(lè)播放器
相關(guān)文章
Android編程計(jì)算函數(shù)時(shí)間戳的相關(guān)方法總結(jié)
這篇文章主要介紹了Android編程計(jì)算函數(shù)時(shí)間戳的相關(guān)方法,結(jié)合實(shí)例形式總結(jié)分析了Android Java、Native、Kernel時(shí)間戳計(jì)算相關(guān)操作技巧,需要的朋友可以參考下2017-05-05Android自定義View實(shí)現(xiàn)可以拖拽的GridView
這篇文章主要為大家詳細(xì)介紹了Android自定義View實(shí)現(xiàn)可以拖拽的GridView,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06Android錄音--AudioRecord、MediaRecorder的使用
本篇文章主要介紹了Android錄音--AudioRecord、MediaRecorder的使用,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2017-02-02Android實(shí)現(xiàn)蝸牛進(jìn)度條效果
這篇文章主要為大家詳細(xì)介紹了Android實(shí)現(xiàn)蝸牛進(jìn)度條效果,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-06-06Android Presentation雙屏異顯開(kāi)發(fā)流程詳細(xì)講解
最近開(kāi)發(fā)的一個(gè)項(xiàng)目,有兩個(gè)屏幕,需要將第二個(gè)頁(yè)面投屏到副屏上,這就需要用到Android的雙屏異顯(Presentation)技術(shù)了,研究了一下,這里做下筆記2023-01-01Android 手勢(shì) 正則匹配圖片實(shí)例代碼
這篇文章主要介紹了Android 手勢(shì) 正則匹配圖片實(shí)例代碼,需要的朋友可以參考下2017-07-07Android自定義控件實(shí)現(xiàn)底部菜單(上)
這篇文章主要為大家詳細(xì)介紹了Android自定義控件實(shí)現(xiàn)底部菜單的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01