1. 程式人生 > >通過ContentProvider播放音樂檔案

通過ContentProvider播放音樂檔案

今天我將為大家分享基於Service與ContentProvider的音樂播放例項,對於接觸Android有一些時日的人來說,Android的核心就是Activity、Service、ContentProvide,BroadcastReceiver,以及串聯它們的Intent五大模組。
Service
Android中的服務,與Activity不同,它是不能與使用者互動的,是執行在後臺的程式。如果我們退出應用時,沒有結束程序,它仍然在後臺執行(當系統記憶體夠用時),那我們什麼時候用到Service呢?比如我們播放音樂的時候,有可能想邊聽音樂邊幹些其他的事情,當我們退出播放音樂的應用時,如果不適用Service,我們就聽不到歌曲了,所以這時候就得用到Service了;又比如當我們一個應用的資料通過網路獲取的,不同時間(一段時間)的資料是不同的,這時候我們可以使用Service在後臺定時更新,而不是每開啟應用的時候再去獲取。

ContentProvider
Android中的內容提供者,它讓我們可以通過一個URI跨應用獲取資料(通常是SQLite資料庫),我覺得Android這個機制還是非常不錯的,特別是我們想獲取Sdcard裡的一些資料時,比如我們想獲取所有Sdcard裡的音訊、視訊、圖片等,我們只要通過一個URI就可以輕鬆搞定,其實我們在開機或者插入Sdcard時,Android會做一些事情,就是它自動建庫,將我們卡里的所有音訊、視訊、圖片資訊等存在相應的表中。
第一步:新建一個Android工程命名為MusicDemo.
第二步:候改main.xml佈局檔案(我這裡增加了四個按鈕,上一首,播放,下一首,暫停)程式碼如下:

    <?xml version="1.0" encoding="utf-8"?>
    <LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
        android:orientation="vertical"
        android:layout_width="fill_parent"
        android:layout_height="fill_parent"
        >
            <TextView  
                android:layout_width
="fill_parent" android:layout_height="wrap_content" android:text="Welcome to Mr Wei's blog." />
<LinearLayout android:orientation="horizontal" android:layout_width="fill_parent" android:layout_height="wrap_content" > <Button android:id="@+id/previous" android:layout_height="fill_parent" android:layout_width="wrap_content" android:layout_weight="1" android:text="上一首" /> <Button android:id="@+id/play" android:layout_height="fill_parent" android:layout_width="wrap_content" android:layout_weight="1" android:text="播放" /> <Button android:id="@+id/next" android:layout_height="fill_parent" android:layout_width="wrap_content" android:layout_weight="1" android:text="下一首" /> <Button android:id="@+id/pause" android:layout_height="fill_parent" android:layout_width="wrap_content" android:layout_weight="1" android:text="暫停" /> </LinearLayout> </LinearLayout>

第三步:新建一個MusicService.java類,播放音樂都是在這個類裡進行的哦,程式碼如下:

    package com.tutor.music;
    import java.io.IOException;
    import android.app.Service;
    import android.content.Intent;
    import android.database.Cursor;
    import android.media.MediaPlayer;
    import android.net.Uri;
    import android.os.IBinder;
    import android.provider.MediaStore;
    import android.widget.Toast;
    public class MusicService extends Service {

            String[] mCursorCols = new String[] {
                            "Audio._id AS _id", // index must match IDCOLIDX below
                            MediaStore.Audio.Media.ARTIST, MediaStore.Audio.Media.ALBUM,
                            MediaStore.Audio.Media.TITLE, MediaStore.Audio.Media.DATA,
                            MediaStore.Audio.Media.MIME_TYPE, MediaStore.Audio.Media.ALBUM_ID,
                            MediaStore.Audio.Media.ARTIST_ID, MediaStore.Audio.Media.DURATION
            };
            private MediaPlayer mMediaPlayer;
            private Cursor mCursor;
            private int mPlayPosition = 0;

            public static final String PLAY_ACTION = "com.tutor.music.PLAY_ACTION";
            public static final String PAUSE_ACTION = "com.tutor.music.PAUSE_ACTION";
            public static final String NEXT_ACTION = "com.tutor.music.NEXT_ACTION";
            public static final String PREVIOUS_ACTION = "com.tutor.music.PREVIOUS_ACTION";
            @Override
            public IBinder onBind(Intent arg0) {
                    // TODO Auto-generated method stub
                    return null;
            }
            @Override
            public void onCreate() {
                    super.onCreate();
                    mMediaPlayer = new MediaPlayer();
                    //通過一個URI可以獲取所有音訊檔案
                    Uri MUSIC_URL = MediaStore.Audio.Media.EXTERNAL_CONTENT_URI;
                    //這裡我過濾了一下,因為我機裡有些音訊檔案是遊戲音訊,很短
                    //播放不到一秒鐘,我這裡作了處理,預設大於10秒的可以看作是歌
                    mCursor = getContentResolver().query(MUSIC_URL, mCursorCols, "duration > 10000", null, null);
            }

            @Override
            public void onStart(Intent intent, int startId) {
                    super.onStart(intent, startId);

                    String action = intent.getAction();
                    if(action.equals(PLAY_ACTION)){
                            play();
                    }else if(action.equals(PAUSE_ACTION)){
                            pause();
                    }else if(action.equals(NEXT_ACTION)){
                            next();
                    }else if(action.equals(PREVIOUS_ACTION)){
                            previous();
                    }
            }

            //play the music
            public void play() {       
                    inite();
            }

            //暫停時,結束服務
            public void pause() {
                    stopSelf();
            }
            //上一首
            public void previous() {
                    if (mPlayPosition == 0) {
                            mPlayPosition = mCursor.getCount() - 1;
                    } else {
                            mPlayPosition--;
                    }
                    inite();
            }
            public void next() {
                    if (mPlayPosition == mCursor.getCount() - 1) {
                            mPlayPosition = 0;
                    } else {
                            mPlayPosition++;
                    }
                    inite();
            }
            public void inite() {
                    mMediaPlayer.reset();
                    String dataSource = getDateByPosition(mCursor, mPlayPosition);
                    String info = getInfoByPosition(mCursor, mPlayPosition);
                    //用Toast顯示歌曲資訊
                    Toast.makeText(getApplicationContext(), info, Toast.LENGTH_SHORT).show();
                    try {
                            mMediaPlayer.setDataSource(dataSource);
                            mMediaPlayer.prepare();
                            mMediaPlayer.start();
                    } catch (IllegalArgumentException e1) {
                            e1.printStackTrace();
                    } catch (IllegalStateException e1) {
                            e1.printStackTrace();
                    } catch (IOException e1) {
                            e1.printStackTrace();
                    }
            }
            //根據位置來獲取歌曲位置
            public String getDateByPosition(Cursor c,int position){
                    c.moveToPosition(position);
                    int dataColumn = c.getColumnIndex(MediaStore.Audio.Media.DATA);               
                    String data = c.getString(dataColumn);
                    return data;
            }
            //獲取當前播放歌曲演唱者及歌名
            public String getInfoByPosition(Cursor c,int position){
                    c.moveToPosition(position);
                    int titleColumn = c.getColumnIndex(MediaStore.Audio.Media.TITLE);
                    int artistColumn = c.getColumnIndex(MediaStore.Audio.Media.ARTIST);
                    String info = c.getString(artistColumn)+" " + c.getString(titleColumn);
                    return info;

            }
            //服務結束時要釋放MediaPlayer
            public void onDestroy() {
                    super.onDestroy();
                    mMediaPlayer.release();
            }
    }

第四步:修改Musicdemo.java程式碼如下(程式碼比較簡潔易懂):

    package com.tutor.music;
    import android.app.Activity;
    import android.content.ComponentName;
    import android.content.Intent;
    import android.os.Bundle;
    import android.view.View;
    import android.view.View.OnClickListener;
    import android.widget.Button;
    public class MusicDemo extends Activity implements OnClickListener {

            private Button mPrevious,mPlay,mNext,mPause;
            private ComponentName component;
        public void onCreate(Bundle savedInstanceState) {
            super.onCreate(savedInstanceState);
            setContentView(R.layout.main);
            //oncreate裡程式碼一如既往的少
            setupViews();
        }
        //初始化一些工作
        public void setupViews(){
                component = new ComponentName(this,
                                    MusicService.class);

                mPrevious = (Button)findViewById(R.id.previous);
                mPlay = (Button)findViewById(R.id.play);
                mNext = (Button)findViewById(R.id.next);
                mPause = (Button)findViewById(R.id.pause);

                mPrevious.setOnClickListener(this);
                mPlay.setOnClickListener(this);
                mNext.setOnClickListener(this);
                mPause.setOnClickListener(this);
        }
        //按鈕點選事件響應
            public void onClick(View v) {
                    if(v == mPrevious){
                            Intent mIntent = new Intent(MusicService.PREVIOUS_ACTION);
                            mIntent.setComponent(component);
                            startService(mIntent);
                    }else if(v == mPlay){
                            Intent mIntent = new Intent(MusicService.PLAY_ACTION);
                            mIntent.setComponent(component);
                            startService(mIntent);
                    }else if(v == mNext){
                            Intent mIntent = new Intent(MusicService.NEXT_ACTION);
                            mIntent.setComponent(component);
                            startService(mIntent);
                    }else{
                            Intent mIntent = new Intent(MusicService.PAUSE_ACTION);
                            mIntent.setComponent(component);
                            startService(mIntent);
                    }

            }
    }

第五步:修改AndroidManifest.xml,這裡只是把我們的MusicService申明進去,不然會報錯(第14行程式碼),程式碼如下:

    <?xml version="1.0" encoding="utf-8"?>
    <manifest xmlns:android="http://schemas.android.com/apk/res/android"
          package="com.tutor.music"
          android:versionCode="1"
          android:versionName="1.0">
        <application android:icon="@drawable/icon" android:Label="@string/app_name">
            <activity android:name=".MusicDemo"
                      android:label="@string/app_name">
                <intent-filter>
                    <action android:name="android.intent.action.MAIN" />
                    <category android:name="android.intent.category.LAUNCHER" />
                </intent-filter>
            </activity>
            <service android:name=".MusicService" android:exported="true" />
        </application>
        <uses-sdk android:minSdkVersion="7" />
    </manifest> 

第六步:執行上述Android工程,效果如下圖所示:效果1:首介面:
效果2:點選播發按鈕開始播放音樂:效果3:我們可以在設定(Settings)->應用(Applications)->正在執行的服務(Running Services)檢視我們啟動了一個新的Service: