Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> Android開發 >> 關於android開發 >> Android學習指南之二十二:實例講解AIDL和遠程Service調用

Android學習指南之二十二:實例講解AIDL和遠程Service調用

編輯:關於android開發

       本講的內容,理解起來很難,也許你看了很多資料也看不明白,但是用起來缺簡單的要命。所以我們干脆拿一個音樂播放器中進度條的實例來說明一下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代碼
  1. <?xml version="1.0" encoding="utf-8"?>  
  2. <manifest package="app.android.elfplayer" xmlns:android="http://schemas.android.com/apk/res/android" android:versioncode="1" android:versionname="1.0">  
  3.         <uses -sdk="" android:minsdkversion="7">  
  4.         <uses -permission="" android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses>  
  5.   
  6.         <application android:label="@string/app_name" android:icon="@drawable/icon">  
  7.                 <activity android:name=".CoverActivity">  
  8.                         <intent -filter="">  
  9.                                 <action android:name="android.intent.action.MAIN">  
  10.                                 <category android:name="android.intent.category.LAUNCHER">  
  11.                         </category></action></intent>  
  12.                 </activity>  
  13.                 <activity android:name=".PlayerActivity">  
  14.                 </activity>  
  15.                 <service android:name=".MusicService" android:enabled="true">  
  16.                 </service>  
  17.         </application>  
  18.   
  19. </uses></manifest>  

       我們注意到有2個Activity,1個Service,還有讀寫外部存儲的權限聲明。

       3、CoverActivity.java的代碼如下:這是個全屏的啟動畫面,2秒後會跳轉到PlayerActivity:

Java代碼
  1. package app.android.elfplayer;   
  2.   
  3. import android.app.Activity;   
  4. import android.content.Intent;   
  5. import android.os.Bundle;   
  6. import android.os.Handler;   
  7. import android.view.Window;   
  8. import android.view.WindowManager;   
  9.   
  10. public class CoverActivity extends Activity {   
  11.         /** Called when the activity is first created. */  
  12.         @Override  
  13.         public void onCreate(Bundle savedInstanceState) {   
  14.                 super.onCreate(savedInstanceState);   
  15.                 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);   
  16.                 requestWindowFeature(Window.FEATURE_NO_TITLE);   
  17.                 setContentView(R.layout.cover);   
  18.   
  19.                 new Handler().postDelayed(new Runnable(){   
  20.   
  21.                  @Override  
  22.                  public void run() {   
  23.                      Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);   
  24.                      CoverActivity.this.startActivity(mainIntent);   
  25.                      CoverActivity.this.finish();   
  26.                  }   
  27.   
  28.                 }, 2000);   
  29.   
  30.         }   
  31. }  

       4、PlayerActivity.java的代碼如下:

Java代碼
  1. package app.android.elfplayer;   
  2.   
  3. import android.app.Activity;   
  4. import android.content.ComponentName;   
  5. import android.content.Context;   
  6. import android.content.Intent;   
  7. import android.content.ServiceConnection;   
  8. import android.os.Bundle;   
  9. import android.os.Handler;   
  10. import android.os.IBinder;   
  11. import android.os.Message;   
  12. import android.os.RemoteException;   
  13. import android.util.Log;   
  14. import android.view.View;   
  15. import android.widget.ImageButton;   
  16. import android.widget.SeekBar;   
  17. import android.widget.SeekBar.OnSeekBarChangeListener;   
  18.   
  19. public class PlayerActivity extends Activity {   
  20.   
  21.         public static final int PLAY = 1;   
  22.         public static final int PAUSE = 2;   
  23.   
  24.         ImageButton imageButtonFavorite;   
  25.         ImageButton imageButtonNext;   
  26.         ImageButton imageButtonPlay;   
  27.         ImageButton imageButtonPre;   
  28.         ImageButton imageButtonRepeat;   
  29.         SeekBar musicSeekBar;   
  30.   
  31.         IServicePlayer iPlayer;   
  32.         boolean isPlaying = false;   
  33.         boolean isLoop = false;           
  34.   
  35.         @Override  
  36.         public void onCreate(Bundle savedInstanceState) {   
  37.                 super.onCreate(savedInstanceState);   
  38.                 setContentView(R.layout.player);   
  39.   
  40.                 imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);   
  41.                 imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);   
  42.                 imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);   
  43.                 imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);   
  44.                 imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);   
  45.                 musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);   
  46.   
  47.                 bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);   
  48.                 startService(new Intent(PlayerActivity.this, MusicService.class));   
  49.   
  50.                 imageButtonPlay.setOnClickListener(new View.OnClickListener() {   
  51.   
  52.                         @Override  
  53.                         public void onClick(View v) {   
  54.                                 Log.i("yao", "imageButtonPlay -> onClick");   
  55.   
  56.                                 if (!isPlaying) {   
  57.                                         try {   
  58.                                                 iPlayer.play();   
  59.                                         } catch (RemoteException e) {   
  60.                                                 e.printStackTrace();   
  61.                                         }   
  62.                                         imageButtonPlay.setBackgroundResource(R.drawable.pause_button);   
  63.                                         isPlaying = true;   
  64.   
  65.                                 } else {   
  66.                                         try {   
  67.                                                 iPlayer.pause();   
  68.                                         } catch (RemoteException e) {   
  69.                                                 e.printStackTrace();   
  70.                                         }   
  71.                                         imageButtonPlay.setBackgroundResource(R.drawable.play_button);   
  72.                                         isPlaying = false;   
  73.                                 }   
  74.                         }   
  75.                 });   
  76.   
  77.                 musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {   
  78.   
  79.                         @Override  
  80.                         public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {   
  81.                         }   
  82.   
  83.                         @Override  
  84.                         public void onStartTrackingTouch(SeekBar seekBar) {   
  85.                         }   
  86.   
  87.                         @Override  
  88.                         public void onStopTrackingTouch(SeekBar seekBar) {   
  89.                                 if (iPlayer != null) {   
  90.                                         try {   
  91.                                                 iPlayer.seekTo(seekBar.getProgress());   
  92.                                         } catch (RemoteException e) {   
  93.                                                 e.printStackTrace();   
  94.                                         }   
  95.                                 }   
  96.                         }   
  97.                 });   
  98.   
  99.                 handler.post(updateThread);   
  100.         }   
  101.   
  102.         private ServiceConnection conn = new ServiceConnection() {   
  103.                 public void onServiceConnected(ComponentName className, IBinder service) {   
  104.                         Log.i("yao", "ServiceConnection -> onServiceConnected");   
  105.                         iPlayer = IServicePlayer.Stub.asInterface(service);   
  106.                 }   
  107.   
  108.                 public void onServiceDisconnected(ComponentName className) {   
  109.                 };   
  110.         };   
  111.   
  112.         Handler handler = new Handler() {   
  113.                 @Override  
  114.                 public void handleMessage(Message msg) {   
  115.                 };   
  116.         };   
  117.   
  118.         private Runnable updateThread = new Runnable() {   
  119.                 @Override  
  120.                 public void run() {   
  121.                         if (iPlayer != null) {   
  122.                                 try {   
  123.                                         musicSeekBar.setMax(iPlayer.getDuration());   
  124.                                         musicSeekBar.setProgress(iPlayer.getCurrentPosition());   
  125.                                 } catch (RemoteException e) {   
  126.                                         e.printStackTrace();   
  127.                                 }   
  128.                         }   
  129.                         handler.post(updateThread);   
  130.                 }   
  131.         };   
  132.   
  133. }  

       5、其中用到的IServicePlayer.aidl,放在和Java文件相同的包中,內容如下:

Java代碼
  1. package app.android.elfplayer;   
  2. interface IServicePlayer{   
  3.         void play();   
  4.         void pause();   
  5.         void stop();   
  6.         int getDuration();   
  7.         int getCurrentPosition();   
  8.         void seekTo(int current);   
  9.         boolean setLoop(boolean loop);   
  10. }  

       一旦你寫好了這個IServicePlayer.aidl文件,ADT會自動幫你在gen目錄下生成IServicePlayer.java文件。

       6、MusicService.java的內容如下:

Java代碼
  1. package app.android.elfplayer;   
  2.   
  3. import android.app.Service;   
  4. import android.content.Intent;   
  5. import android.media.MediaPlayer;   
  6. import android.os.IBinder;   
  7. import android.os.RemoteException;   
  8. import android.util.Log;   
  9.   
  10. public class MusicService extends Service {   
  11.   
  12.         String tag = "yao";   
  13.   
  14.         public static MediaPlayer mPlayer;   
  15.   
  16.         public boolean isPause = false;   
  17.   
  18.         IServicePlayer.Stub stub = new IServicePlayer.Stub() {   
  19.   
  20.                 @Override  
  21.                 public void play() throws RemoteException {   
  22.                         mPlayer.start();   
  23.                 }   
  24.   
  25.                 @Override  
  26.                 public void pause() throws RemoteException {   
  27.                         mPlayer.pause();   
  28.                 }   
  29.   
  30.                 @Override  
  31.                 public void stop() throws RemoteException {   
  32.                         mPlayer.stop();   
  33.                 }   
  34.   
  35.                 @Override  
  36.                 public int getDuration() throws RemoteException {   
  37.                         return mPlayer.getDuration();   
  38.                 }   
  39.   
  40.                 @Override  
  41.                 public int getCurrentPosition() throws RemoteException {   
  42.                         return mPlayer.getCurrentPosition();   
  43.                 }   
  44.   
  45.                 @Override  
  46.                 public void seekTo(int current) throws RemoteException {   
  47.                         mPlayer.seekTo(current);   
  48.                 }   
  49.   
  50.                 @Override  
  51.                 public boolean setLoop(boolean loop) throws RemoteException {   
  52.                         return false;   
  53.                 }   
  54.   
  55.         };   
  56.   
  57.         @Override  
  58.         public void onCreate() {   
  59.                 Log.i(tag, "MusicService onCreate()");   
  60.                 mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));   
  61.         }   
  62.   
  63.         @Override  
  64.         public IBinder onBind(Intent intent) {   
  65.                 return stub;   
  66.         }   
  67.   
  68. }  

       7、編譯並運行程序,查看結果:

Android學習指南之二十二:實例講解AIDL和遠程Service調用

Android學習指南之二十二:實例講解AIDL和遠程Service調用

       最後總結一下,AIDL提供了一種非常簡單的方式,讓我們可以把一個進程內的對象或方法暴露給另一個程序使用,就好象另一個程序也擁有這些功能一樣。最後感謝一首歌這個網站,本講的圖片素材采用的是他們的UI元素,好了,本講就到這裡。

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