本講的內容,理解起來很難,也許你看了很多資料也看不明白,但是用起來缺簡單的要命。所以我們干脆拿一個音樂播放器中進度條的實例來說明一下AIDL和Remote Service的價值和使用方法,你把這個例子跑一遍,體會一下就OK了。下面的例子是我正在准備的項目實例中的一部分。
簡要分析
首先說明一下我們面臨的問題,如果看不懂下面的描述請看前面的課程:
第一,我們知道在AndroId中如果需要進行音樂播放,最方面的方法就是使用自帶的MediaPlayer對象,如果我們在Activity中控制MediaPlayer對象進行播放,那麼一旦你打開了另外一個程序譬如浏覽器,那麼歌聲就會立刻停止,這當然不是我們需要的結果。 我們需要的是在做其他事情的同時能夠在後台聽歌,於是我們就需要把對MediaPlayer對象的操作放在後台Service中去。
第二,我們已經把對MediaPlayer的操作轉移到Service中去了,按照我們以前的做法,我們在Activity中發送一個Intent對象給Service對象,在Intent中傳送播放啊、暫停啊一類的信息給Service,這樣Service就知道該怎麼做了。這一切看起來很美好,可是現在出了一個新問題,那就是我想在Activity中顯示一個進度條,這個進度條要跟著Service中的MediaPlayer中的歌曲進度同步向前走,而且如果我點擊進度條中的某一個位置,還想讓歌曲跳轉到新的時間點繼續播放,這個,該怎麼實現?
第三,我們需要在Activity中操作Service中的MediaPlayer對象,就好像這個對象是自己的一樣。我們可以采用Android接口定義語言AIDL(Android Interface Definition Language)技術:
1、把Service中針對MediaPlayer的操作封裝成一個接口(.aidl文件) 。
2、在Service中建個子類實現這接口的存根(stub)對象。
3、並在onBind()方法中返回這個存根對象。
4、在Activity中使用綁定服務的方式連接Service,但是不用Intent來傳遞信息,而是在ServiceConnection的onServiceConnected方法裡,獲得Service中Stub對象的客戶端使用代理。我們通過操作Activity中的代理就可以達到操作Service中的MediaPlayer對象的目的。這樣我們就可以想用本地對象一樣操作Service中的對象了,那麼進度條一類的需求自然也就迎刃而解。
開發實例
下面的例子,並不是專門為本講准備的,所以有些無關代碼,而且沒加注釋,請見諒:
1、新建一個項目App_elfPlayer ,啟動Activity是個啟動畫面:CoverActivity。
2、AndroidManifest.xml 的內容如下:
XML/HTML代碼
- <?xml version="1.0" encoding="utf-8"?>
- <manifest package="app.android.elfplayer" xmlns:android="http://schemas.android.com/apk/res/android" android:versioncode="1" android:versionname="1.0">
- <uses -sdk="" android:minsdkversion="7">
- <uses -permission="" android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses>
-
- <application android:label="@string/app_name" android:icon="@drawable/icon">
- <activity android:name=".CoverActivity">
- <intent -filter="">
- <action android:name="android.intent.action.MAIN">
- <category android:name="android.intent.category.LAUNCHER">
- </category></action></intent>
- </activity>
- <activity android:name=".PlayerActivity">
- </activity>
- <service android:name=".MusicService" android:enabled="true">
- </service>
- </application>
-
- </uses></manifest>
我們注意到有2個Activity,1個Service,還有讀寫外部存儲的權限聲明。
3、CoverActivity.java的代碼如下:這是個全屏的啟動畫面,2秒後會跳轉到PlayerActivity:
Java代碼
- package app.android.elfplayer;
-
- import android.app.Activity;
- import android.content.Intent;
- import android.os.Bundle;
- import android.os.Handler;
- import android.view.Window;
- import android.view.WindowManager;
-
- public class CoverActivity extends Activity {
- /** Called when the activity is first created. */
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
- requestWindowFeature(Window.FEATURE_NO_TITLE);
- setContentView(R.layout.cover);
-
- new Handler().postDelayed(new Runnable(){
-
- @Override
- public void run() {
- Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);
- CoverActivity.this.startActivity(mainIntent);
- CoverActivity.this.finish();
- }
-
- }, 2000);
-
- }
- }
4、PlayerActivity.java的代碼如下:
Java代碼
- package app.android.elfplayer;
-
- import android.app.Activity;
- import android.content.ComponentName;
- import android.content.Context;
- import android.content.Intent;
- import android.content.ServiceConnection;
- import android.os.Bundle;
- import android.os.Handler;
- import android.os.IBinder;
- import android.os.Message;
- import android.os.RemoteException;
- import android.util.Log;
- import android.view.View;
- import android.widget.ImageButton;
- import android.widget.SeekBar;
- import android.widget.SeekBar.OnSeekBarChangeListener;
-
- public class PlayerActivity extends Activity {
-
- public static final int PLAY = 1;
- public static final int PAUSE = 2;
-
- ImageButton imageButtonFavorite;
- ImageButton imageButtonNext;
- ImageButton imageButtonPlay;
- ImageButton imageButtonPre;
- ImageButton imageButtonRepeat;
- SeekBar musicSeekBar;
-
- IServicePlayer iPlayer;
- boolean isPlaying = false;
- boolean isLoop = false;
-
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.player);
-
- imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);
- imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);
- imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);
- imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);
- imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);
- musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);
-
- bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);
- startService(new Intent(PlayerActivity.this, MusicService.class));
-
- imageButtonPlay.setOnClickListener(new View.OnClickListener() {
-
- @Override
- public void onClick(View v) {
- Log.i("yao", "imageButtonPlay -> onClick");
-
- if (!isPlaying) {
- try {
- iPlayer.play();
- } catch (RemoteException e) {
- e.printStackTrace();
- }
- imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
- isPlaying = true;
-
- } else {
- try {
- iPlayer.pause();
- } catch (RemoteException e) {
- e.printStackTrace();
- }
- imageButtonPlay.setBackgroundResource(R.drawable.play_button);
- isPlaying = false;
- }
- }
- });
-
- musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {
-
- @Override
- public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
- }
-
- @Override
- public void onStartTrackingTouch(SeekBar seekBar) {
- }
-
- @Override
- public void onStopTrackingTouch(SeekBar seekBar) {
- if (iPlayer != null) {
- try {
- iPlayer.seekTo(seekBar.getProgress());
- } catch (RemoteException e) {
- e.printStackTrace();
- }
- }
- }
- });
-
- handler.post(updateThread);
- }
-
- private ServiceConnection conn = new ServiceConnection() {
- public void onServiceConnected(ComponentName className, IBinder service) {
- Log.i("yao", "ServiceConnection -> onServiceConnected");
- iPlayer = IServicePlayer.Stub.asInterface(service);
- }
-
- public void onServiceDisconnected(ComponentName className) {
- };
- };
-
- Handler handler = new Handler() {
- @Override
- public void handleMessage(Message msg) {
- };
- };
-
- private Runnable updateThread = new Runnable() {
- @Override
- public void run() {
- if (iPlayer != null) {
- try {
- musicSeekBar.setMax(iPlayer.getDuration());
- musicSeekBar.setProgress(iPlayer.getCurrentPosition());
- } catch (RemoteException e) {
- e.printStackTrace();
- }
- }
- handler.post(updateThread);
- }
- };
-
- }
5、其中用到的IServicePlayer.aidl,放在和Java文件相同的包中,內容如下:
Java代碼
- package app.android.elfplayer;
- interface IServicePlayer{
- void play();
- void pause();
- void stop();
- int getDuration();
- int getCurrentPosition();
- void seekTo(int current);
- boolean setLoop(boolean loop);
- }
一旦你寫好了這個IServicePlayer.aidl文件,ADT會自動幫你在gen目錄下生成IServicePlayer.java文件。
6、MusicService.java的內容如下:
Java代碼
- package app.android.elfplayer;
-
- import android.app.Service;
- import android.content.Intent;
- import android.media.MediaPlayer;
- import android.os.IBinder;
- import android.os.RemoteException;
- import android.util.Log;
-
- public class MusicService extends Service {
-
- String tag = "yao";
-
- public static MediaPlayer mPlayer;
-
- public boolean isPause = false;
-
- IServicePlayer.Stub stub = new IServicePlayer.Stub() {
-
- @Override
- public void play() throws RemoteException {
- mPlayer.start();
- }
-
- @Override
- public void pause() throws RemoteException {
- mPlayer.pause();
- }
-
- @Override
- public void stop() throws RemoteException {
- mPlayer.stop();
- }
-
- @Override
- public int getDuration() throws RemoteException {
- return mPlayer.getDuration();
- }
-
- @Override
- public int getCurrentPosition() throws RemoteException {
- return mPlayer.getCurrentPosition();
- }
-
- @Override
- public void seekTo(int current) throws RemoteException {
- mPlayer.seekTo(current);
- }
-
- @Override
- public boolean setLoop(boolean loop) throws RemoteException {
- return false;
- }
-
- };
-
- @Override
- public void onCreate() {
- Log.i(tag, "MusicService onCreate()");
- mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));
- }
-
- @Override
- public IBinder onBind(Intent intent) {
- return stub;
- }
-
- }
7、編譯並運行程序,查看結果:
最後總結一下,AIDL提供了一種非常簡單的方式,讓我們可以把一個進程內的對象或方法暴露給另一個程序使用,就好象另一個程序也擁有這些功能一樣。最後感謝一首歌這個網站,本講的圖片素材采用的是他們的UI元素,好了,本講就到這裡。