AIDL和远程Service调用

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


lesson22_aidl

本讲内容: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<?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

01package app.android.elfplayer;
02 
03import android.app.Activity;
04import android.content.Intent;
05import android.os.Bundle;
06import android.os.Handler;
07import android.view.Window;
08import android.view.WindowManager;
09 
10public class CoverActivity extendsActivity {
11/** Called when the activity is first created. */
12@Override
13publicvoid onCreate(Bundle savedInstanceState) {
14super.onCreate(savedInstanceState);
15getWindow().setFlags(WindowManager.LayoutParams.FLAG_FULLSCREEN, WindowManager.LayoutParams.FLAG_FULLSCREEN);
16requestWindowFeature(Window.FEATURE_NO_TITLE);
17setContentView(R.layout.cover);
18
19newHandler().postDelayed(newRunnable(){
20
21@Override
22publicvoid run() {
23Intent mainIntent =new Intent(CoverActivity.this,PlayerActivity.class);
24CoverActivity.this.startActivity(mainIntent);
25CoverActivity.this.finish();
26}
27
28}, 2000);
29
30}
31}

4、PlayerActivity.java的代码如下:

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

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

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

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

6、MusicService.java的内容如下:

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

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

image

image

image

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

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

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

24 Mar

lesson33_customui

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

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

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

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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值