本讲内容: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 的内容如下:
01 |
<? xml version = "1.0" encoding = "utf-8" ?> |
02 |
< manifest package = "app.android.elfplayer" xmlns:android = "http://schemas.android.com/apk/res/android" android:versioncode = "1" android:versionname = "1.0" > |
03 |
< uses -sdk = "" android:minsdkversion = "7" > |
04 |
< uses -permission = "" android:name = "android.permission.WRITE_EXTERNAL_STORAGE" ></ uses > |
05 |
06 |
< application android:label = "@string/app_name" android:icon = "@drawable/icon" > |
07 |
< activity android:name = ".CoverActivity" > |
08 |
< intent -filter = "" > |
09 |
< 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
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 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的代码如下:
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 extends Activity { |
020 |
021 |
public static final int PLAY = 1 ; |
022 |
public static 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 |
boolean isPlaying = false ; |
033 |
boolean isLoop = false ; |
034 |
035 |
@Override |
036 |
public void 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( new Intent(PlayerActivity. this , MusicService. class ), conn, Context.BIND_AUTO_CREATE); |
048 |
startService( new Intent(PlayerActivity. this , MusicService. class )); |
049 |
050 |
imageButtonPlay.setOnClickListener( new View.OnClickListener() { |
051 |
052 |
@Override |
053 |
public void 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( new OnSeekBarChangeListener() { |
078 |
079 |
@Override |
080 |
public void onProgressChanged(SeekBar seekBar, int progress, boolean fromUser) { |
081 |
} |
082 |
083 |
@Override |
084 |
public void onStartTrackingTouch(SeekBar seekBar) { |
085 |
} |
086 |
087 |
@Override |
088 |
public void 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 |
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文件相同的包中,内容如下:
01 |
package app.android.elfplayer; |
02 |
interface IServicePlayer{ |
03 |
void play(); |
04 |
void pause(); |
05 |
void stop(); |
06 |
int getDuration(); |
07 |
int getCurrentPosition(); |
08 |
void seekTo( int current); |
09 |
boolean setLoop( boolean loop); |
10 |
} |
一旦你写好了这个IServicePlayer.aidl文件,ADT会自动帮你在gen目录下生成IServicePlayer.java文件
6、MusicService.java的内容如下:
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 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、其它代码和资源可以参见本讲附带的源代码,编译并运行程序,查看结果:
最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。
最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素,好了,本讲就到这里。