第二十二讲:AIDL和远程Service调用

本讲内容:AIDL和远程Service调用
本讲源代码:App_elfPlayer.zip
本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下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 的内容如下:
  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.         <application android:label="@string/app_name" android:icon="@drawable/icon">
  6.                 <activity android:name=".CoverActivity">
  7.                         <intent -filter="">
  8.                                 <action android:name="android.intent.action.MAIN">
  9.                                 <category android:name="android.intent.category.LAUNCHER">
  10.                         </category></action></intent>
  11.                 </activity>
  12.                 <activity android:name=".PlayerActivity">
  13.                 </activity>
  14.                 <service android:name=".MusicService" android:enabled="true">
  15.                 </service>
  16.         </application>

  17. </uses></manifest>

我们注意到有2个Activity,1个Service,还有读写外部存储的权限声明。

3、CoverActivity.java的代码如下:

这是个全屏的启动画面,2秒后会跳转到PlayerActivity。
  1. package app.android.elfplayer;

  2. import android.app.Activity;
  3. import android.content.Intent;
  4. import android.os.Bundle;
  5. import android.os.Handler;
  6. import android.view.Window;
  7. import android.view.WindowManager;

  8. public class CoverActivity extends Activity {
  9.         /** Called when the activity is first created. */
  10.         @Override
  11.         public void onCreate(Bundle savedInstanceState) {
  12.                 super.onCreate(savedInstanceState);
  13.                 getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
  14.                 requestWindowFeature(Window.FEATURE_NO_TITLE);
  15.                 setContentView(R.layout.cover);

  16.                 new Handler().postDelayed(new Runnable(){

  17.                  @Override
  18.                  public void run() {
  19.                      Intent mainIntent = new Intent(CoverActivity.this,PlayerActivity.class);
  20.                      CoverActivity.this.startActivity(mainIntent);
  21.                      CoverActivity.this.finish();
  22.                  }

  23.                 }, 2000);

  24.         }
  25. }

4、PlayerActivity.java的代码如下:
  1. package app.android.elfplayer;

  2. import android.app.Activity;
  3. import android.content.ComponentName;
  4. import android.content.Context;
  5. import android.content.Intent;
  6. import android.content.ServiceConnection;
  7. import android.os.Bundle;
  8. import android.os.Handler;
  9. import android.os.IBinder;
  10. import android.os.Message;
  11. import android.os.RemoteException;
  12. import android.util.Log;
  13. import android.view.View;
  14. import android.widget.ImageButton;
  15. import android.widget.SeekBar;
  16. import android.widget.SeekBar.OnSeekBarChangeListener;

  17. public class PlayerActivity extends Activity {

  18.         public static final int PLAY = 1;
  19.         public static final int PAUSE = 2;

  20.         ImageButton imageButtonFavorite;
  21.         ImageButton imageButtonNext;
  22.         ImageButton imageButtonPlay;
  23.         ImageButton imageButtonPre;
  24.         ImageButton imageButtonRepeat;
  25.         SeekBar musicSeekBar;

  26.         IServicePlayer iPlayer;
  27.         boolean isPlaying = false;
  28.         boolean isLoop = false;       

  29.         @Override
  30.         public void onCreate(Bundle savedInstanceState) {
  31.                 super.onCreate(savedInstanceState);
  32.                 setContentView(R.layout.player);

  33.                 imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);
  34.                 imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);
  35.                 imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);
  36.                 imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);
  37.                 imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);
  38.                 musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);

  39.                 bindService(new Intent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);
  40.                 startService(new Intent(PlayerActivity.this, MusicService.class));

  41.                 imageButtonPlay.setOnClickListener(new View.OnClickListener() {

  42.                         @Override
  43.                         public void onClick(View v) {
  44.                                 Log.i("yao", "imageButtonPlay -> onClick");

  45.                                 if (!isPlaying) {
  46.                                         try {
  47.                                                 iPlayer.play();
  48.                                         } catch (RemoteException e) {
  49.                                                 e.printStackTrace();
  50.                                         }
  51.                                         imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
  52.                                         isPlaying = true;

  53.                                 } else {
  54.                                         try {
  55.                                                 iPlayer.pause();
  56.                                         } catch (RemoteException e) {
  57.                                                 e.printStackTrace();
  58.                                         }
  59.                                         imageButtonPlay.setBackgroundResource(R.drawable.play_button);
  60.                                         isPlaying = false;
  61.                                 }
  62.                         }
  63.                 });

  64.                 musicSeekBar.setOnSeekBarChangeListener(new OnSeekBarChangeListener() {

  65.                         @Override
  66.                         public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) {
  67.                         }

  68.                         @Override
  69.                         public void onStartTrackingTouch(SeekBar seekBar) {
  70.                         }

  71.                         @Override
  72.                         public void onStopTrackingTouch(SeekBar seekBar) {
  73.                                 if (iPlayer != null) {
  74.                                         try {
  75.                                                 iPlayer.seekTo(seekBar.getProgress());
  76.                                         } catch (RemoteException e) {
  77.                                                 e.printStackTrace();
  78.                                         }
  79.                                 }
  80.                         }
  81.                 });

  82.                 handler.post(updateThread);
  83.         }

  84.         private ServiceConnection conn = new ServiceConnection() {
  85.                 public void onServiceConnected(ComponentName className, IBinder service) {
  86.                         Log.i("yao", "ServiceConnection -> onServiceConnected");
  87.                         iPlayer = IServicePlayer.Stub.asInterface(service);
  88.                 }

  89.                 public void onServiceDisconnected(ComponentName className) {
  90.                 };
  91.         };

  92.         Handler handler = new Handler() {
  93.                 @Override
  94.                 public void handleMessage(Message msg) {
  95.                 };
  96.         };

  97.         private Runnable updateThread = new Runnable() {
  98.                 @Override
  99.                 public void run() {
  100.                         if (iPlayer != null) {
  101.                                 try {
  102.                                         musicSeekBar.setMax(iPlayer.getDuration());
  103.                                         musicSeekBar.setProgress(iPlayer.getCurrentPosition());
  104.                                 } catch (RemoteException e) {
  105.                                         e.printStackTrace();
  106.                                 }
  107.                         }
  108.                         handler.post(updateThread);
  109.                 }
  110.         };

  111. }

5、其中用到的IServicePlayer.aidl,放在和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的内容如下:
  1. package app.android.elfplayer;

  2. import android.app.Service;
  3. import android.content.Intent;
  4. import android.media.MediaPlayer;
  5. import android.os.IBinder;
  6. import android.os.RemoteException;
  7. import android.util.Log;

  8. public class MusicService extends Service {

  9.         String tag = "yao";

  10.         public static MediaPlayer mPlayer;

  11.         public boolean isPause = false;

  12.         IServicePlayer.Stub stub = new IServicePlayer.Stub() {

  13.                 @Override
  14.                 public void play() throws RemoteException {
  15.                         mPlayer.start();
  16.                 }

  17.                 @Override
  18.                 public void pause() throws RemoteException {
  19.                         mPlayer.pause();
  20.                 }

  21.                 @Override
  22.                 public void stop() throws RemoteException {
  23.                         mPlayer.stop();
  24.                 }

  25.                 @Override
  26.                 public int getDuration() throws RemoteException {
  27.                         return mPlayer.getDuration();
  28.                 }

  29.                 @Override
  30.                 public int getCurrentPosition() throws RemoteException {
  31.                         return mPlayer.getCurrentPosition();
  32.                 }

  33.                 @Override
  34.                 public void seekTo(int current) throws RemoteException {
  35.                         mPlayer.seekTo(current);
  36.                 }

  37.                 @Override
  38.                 public boolean setLoop(boolean loop) throws RemoteException {
  39.                         return false;
  40.                 }

  41.         };

  42.         @Override
  43.         public void onCreate() {
  44.                 Log.i(tag, "MusicService onCreate()");
  45.                 mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));
  46.         }

  47.         @Override
  48.         public IBinder onBind(Intent intent) {
  49.                 return stub;
  50.         }

  51. }

7、其它代码和资源可以参见本讲附带的源代码,编译并运行程序,查看结果:

3.png (400.35 KB, 下载次数: 273)

下载附件 保存到相册

2011-4-24 20:38 上传


最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素,好了,本讲就到这里。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值