AIDL和远程Service调用

http://android.yaohuiji.com/archives/category/android-basic


本讲内容:AIDL和远程Service调用

本讲源代码:App_elfPlayer

本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下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 的内容如下:

view source print ?
01 <?xmlversion="1.0"encoding="utf-8"?>
02 <manifestandroid:versionname="1.0"android:versioncode="1"xmlns:android="http://schemas.android.com/apk/res/android"package="app.android.elfplayer">
03 <usesandroid:minsdkversion="7"-sdk="">
04 <usesandroid:name="android.permission.WRITE_EXTERNAL_STORAGE"-permission=""></uses>
05  
06 <applicationandroid:icon="@drawable/icon"android:label="@string/app_name">
07 <activityandroid:name=".CoverActivity">
08 <intent-filter="">
09 <actionandroid:name="android.intent.action.MAIN">
10 <categoryandroid:name="android.intent.category.LAUNCHER">
11 </category></action></intent>
12 </activity>
13 <activityandroid:name=".PlayerActivity">
14 </activity>
15 <serviceandroid:name=".MusicService"android:enabled="true">
16 </service>
17 </application>
18  
19 </uses></manifest>

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

3、CoverActivity.java的代码如下:这是个全屏的启动画面,2秒后会跳转到PlayerActivity

view source print ?
01 package app.android.elfplayer;
02  
03 import android.app.Activity;
04 import android.content.Intent;
05 import android.os.Bundle;
06 import android.os.Handler;
07 import android.view.Window;
08 import android.view.WindowManager;
09  
10 public class CoverActivity extendsActivity {
11 /** Called when the activity is first created. */
12 @Override
13 publicvoid 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 newHandler().postDelayed(newRunnable(){
20
21 @Override
22 publicvoid 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的代码如下:

view source print ?
001 package app.android.elfplayer;
002  
003 import android.app.Activity;
004 import android.content.ComponentName;
005 import android.content.Context;
006 import android.content.Intent;
007 import android.content.ServiceConnection;
008 import android.os.Bundle;
009 import android.os.Handler;
010 import android.os.IBinder;
011 import android.os.Message;
012 import android.os.RemoteException;
013 import android.util.Log;
014 import android.view.View;
015 import android.widget.ImageButton;
016 import android.widget.SeekBar;
017 import android.widget.SeekBar.OnSeekBarChangeListener;
018  
019 public class PlayerActivity extendsActivity {
020  
021 publicstatic final int PLAY = 1;
022 publicstatic final int PAUSE = 2;
023  
024 ImageButton imageButtonFavorite;
025 ImageButton imageButtonNext;
026 ImageButton imageButtonPlay;
027 ImageButton imageButtonPre;
028 ImageButton imageButtonRepeat;
029 SeekBar musicSeekBar;
030  
031 IServicePlayer iPlayer;
032 booleanisPlaying = false;
033 booleanisLoop = false;
034  
035 @Override
036 publicvoid onCreate(Bundle savedInstanceState) {
037 super.onCreate(savedInstanceState);
038 setContentView(R.layout.player);
039  
040 imageButtonFavorite = (ImageButton) findViewById(R.id.imageButtonFavorite);
041 imageButtonNext = (ImageButton) findViewById(R.id.imageButtonNext);
042 imageButtonPlay = (ImageButton) findViewById(R.id.imageButtonPlay);
043 imageButtonPre = (ImageButton) findViewById(R.id.imageButtonPre);
044 imageButtonRepeat = (ImageButton) findViewById(R.id.imageButtonRepeat);
045 musicSeekBar = (SeekBar) findViewById(R.id.musicSeekBar);
046  
047 bindService(newIntent(PlayerActivity.this, MusicService.class), conn, Context.BIND_AUTO_CREATE);
048 startService(newIntent(PlayerActivity.this, MusicService.class));
049  
050 imageButtonPlay.setOnClickListener(newView.OnClickListener() {
051  
052 @Override
053 publicvoid onClick(View v) {
054 Log.i("yao","imageButtonPlay -> onClick");
055  
056 if(!isPlaying) {
057 try{
058 iPlayer.play();
059 } catch (RemoteException e) {
060 e.printStackTrace();
061 }
062 imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
063 isPlaying =true;
064  
065 } else {
066 try{
067 iPlayer.pause();
068 } catch (RemoteException e) {
069 e.printStackTrace();
070 }
071 imageButtonPlay.setBackgroundResource(R.drawable.play_button);
072 isPlaying =false;
073 }
074 }
075 });
076  
077 musicSeekBar.setOnSeekBarChangeListener(newOnSeekBarChangeListener() {
078  
079 @Override
080 publicvoid onProgressChanged(SeekBar seekBar,int progress, boolean fromUser) {
081 }
082  
083 @Override
084 publicvoid onStartTrackingTouch(SeekBar seekBar) {
085 }
086  
087 @Override
088 publicvoid onStopTrackingTouch(SeekBar seekBar) {
089 if(iPlayer != null) {
090 try{
091 iPlayer.seekTo(seekBar.getProgress());
092 } catch (RemoteException e) {
093 e.printStackTrace();
094 }
095 }
096 }
097 });
098
099 handler.post(updateThread);
100 }
101  
102 privateServiceConnection conn = newServiceConnection() {
103 publicvoid onServiceConnected(ComponentName className, IBinder service) {
104 Log.i("yao","ServiceConnection -> onServiceConnected");
105 iPlayer = IServicePlayer.Stub.asInterface(service);
106 }
107  
108 publicvoid onServiceDisconnected(ComponentName className) {
109 };
110 };
111  
112 Handler handler =new Handler() {
113 @Override
114 publicvoid handleMessage(Message msg) {
115 };
116 };
117  
118 privateRunnable updateThread = newRunnable() {
119 @Override
120 publicvoid 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文件相同的包中,内容如下:

view source print ?
01 package app.android.elfplayer;
02 interface IServicePlayer{
03 voidplay();
04 voidpause();
05 voidstop();
06 intgetDuration();
07 intgetCurrentPosition();
08 voidseekTo(int current);
09 booleansetLoop(booleanloop);
10 }

一旦你写好了这个IServicePlayer.aidl文件,ADT会自动帮你在gen目录下生成IServicePlayer.java文件

6、MusicService.java的内容如下:

view source print ?
01 package app.android.elfplayer;
02  
03 import android.app.Service;
04 import android.content.Intent;
05 import android.media.MediaPlayer;
06 import android.os.IBinder;
07 import android.os.RemoteException;
08 import android.util.Log;
09  
10 public class MusicService extendsService {
11
12 String tag ="yao";
13  
14 publicstatic MediaPlayer mPlayer;
15  
16 publicboolean isPause = false;
17  
18 IServicePlayer.Stub stub =new IServicePlayer.Stub() {
19
20 @Override
21 publicvoid play() throwsRemoteException {
22 mPlayer.start();
23 }
24  
25 @Override
26 publicvoid pause() throwsRemoteException {
27 mPlayer.pause();
28 }
29  
30 @Override
31 publicvoid stop() throwsRemoteException {
32 mPlayer.stop();
33 }
34  
35 @Override
36 publicint getDuration() throws RemoteException {
37 returnmPlayer.getDuration();
38 }
39  
40 @Override
41 publicint getCurrentPosition()throws RemoteException {
42 returnmPlayer.getCurrentPosition();
43 }
44  
45 @Override
46 publicvoid seekTo(intcurrent) throwsRemoteException {
47 mPlayer.seekTo(current);
48 }
49  
50 @Override
51 publicboolean setLoop(booleanloop) throws RemoteException {
52 returnfalse;
53 }
54  
55 };
56  
57 @Override
58 publicvoid onCreate() {
59 Log.i(tag,"MusicService onCreate()");
60 mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD("wind.mp3"));
61 }
62  
63 @Override
64 publicIBinder onBind(Intent intent) {
65 returnstub;
66 }
67  
68 }

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

最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。

最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素,好了,本讲就到这里。

11 Comments

Posted in 1.Android基础篇

第三十三讲:自定义Android UI组件

24 Mar

本讲内容:学会自定义属性、自定义UI组件

下午方便的话,下午就填上。

往前提一下,准备填这个坑。 2011-03-24

14 Comments

Posted in 1.Android基础篇

你可能感兴趣的:(AIDL和远程Service调用)