本讲内容: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 >
|
06 |
< application
android:label = "@string/app_name"
android:icon = "@drawable/icon" >
|
07 |
< activity
android:name = ".CoverActivity" >
|
09 |
< action
android:name = "android.intent.action.MAIN" >
|
10 |
< category
android:name = "android.intent.category.LAUNCHER" >
|
11 |
</ category ></ action ></ intent >
|
13 |
< activity
android:name = ".PlayerActivity" >
|
15 |
< service
android:name = ".MusicService"
android:enabled = "true" >
|
我们注意到有2个Activity,1个Service,还有读写外部存储的权限声明
3、CoverActivity.java的代码如下:这是个全屏的启动画面,2秒后会跳转到PlayerActivity
01 |
package app.android.elfplayer;
|
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;
|
10 |
public class
CoverActivity extends
Activity {
|
11 |
/** Called when the activity is first created. */
|
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);
|
19 |
new
Handler().postDelayed( new
Runnable(){
|
23 |
Intent mainIntent =
new Intent(CoverActivity. this ,PlayerActivity. class );
|
24 |
CoverActivity. this .startActivity(mainIntent);
|
25 |
CoverActivity. this .finish();
|
4、PlayerActivity.java的代码如下:
001 |
package app.android.elfplayer;
|
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;
|
019 |
public class
PlayerActivity extends
Activity {
|
021 |
public
static final
int PLAY = 1 ;
|
022 |
public
static final
int PAUSE = 2 ;
|
024 |
ImageButton imageButtonFavorite;
|
025 |
ImageButton imageButtonNext;
|
026 |
ImageButton imageButtonPlay;
|
027 |
ImageButton imageButtonPre;
|
028 |
ImageButton imageButtonRepeat;
|
029 |
SeekBar musicSeekBar;
|
031 |
IServicePlayer iPlayer;
|
032 |
boolean
isPlaying = false ;
|
033 |
boolean
isLoop = false ;
|
036 |
public
void onCreate(Bundle savedInstanceState) {
|
037 |
super .onCreate(savedInstanceState);
|
038 |
setContentView(R.layout.player);
|
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);
|
047 |
bindService( new
Intent(PlayerActivity. this , MusicService. class ), conn, Context.BIND_AUTO_CREATE);
|
048 |
startService( new
Intent(PlayerActivity. this , MusicService. class ));
|
050 |
imageButtonPlay.setOnClickListener( new
View.OnClickListener() {
|
053 |
public
void onClick(View v) {
|
054 |
Log.i( "yao" ,
"imageButtonPlay -> onClick" );
|
059 |
}
catch (RemoteException e) {
|
062 |
imageButtonPlay.setBackgroundResource(R.drawable.pause_button);
|
068 |
}
catch (RemoteException e) {
|
071 |
imageButtonPlay.setBackgroundResource(R.drawable.play_button);
|
077 |
musicSeekBar.setOnSeekBarChangeListener( new
OnSeekBarChangeListener() {
|
080 |
public
void onProgressChanged(SeekBar seekBar,
int progress, boolean
fromUser) {
|
084 |
public
void onStartTrackingTouch(SeekBar seekBar) {
|
088 |
public
void onStopTrackingTouch(SeekBar seekBar) {
|
089 |
if
(iPlayer != null ) {
|
091 |
iPlayer.seekTo(seekBar.getProgress());
|
092 |
}
catch (RemoteException e) {
|
099 |
handler.post(updateThread);
|
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);
|
108 |
public
void onServiceDisconnected(ComponentName className) {
|
112 |
Handler handler =
new Handler() {
|
114 |
public
void handleMessage(Message msg) {
|
118 |
private
Runnable updateThread = new
Runnable() {
|
121 |
if
(iPlayer != null ) {
|
123 |
musicSeekBar.setMax(iPlayer.getDuration());
|
124 |
musicSeekBar.setProgress(iPlayer.getCurrentPosition());
|
125 |
}
catch (RemoteException e) {
|
129 |
handler.post(updateThread);
|
5、其中用到的IServicePlayer.aidl,放在和Java文件相同的包中,内容如下:
01 |
package app.android.elfplayer;
|
02 |
interface IServicePlayer{
|
07 |
int
getCurrentPosition();
|
08 |
void
seekTo( int
current);
|
09 |
boolean
setLoop( boolean
loop);
|
一旦你写好了这个IServicePlayer.aidl文件,ADT会自动帮你在gen目录下生成IServicePlayer.java文件
6、MusicService.java的内容如下:
01 |
package app.android.elfplayer;
|
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;
|
10 |
public class
MusicService extends
Service {
|
14 |
public
static MediaPlayer mPlayer;
|
16 |
public
boolean isPause = false ;
|
18 |
IServicePlayer.Stub stub =
new IServicePlayer.Stub() {
|
21 |
public
void play() throws
RemoteException {
|
26 |
public
void pause() throws
RemoteException {
|
31 |
public
void stop() throws
RemoteException {
|
36 |
public
int getDuration() throws
RemoteException {
|
37 |
return
mPlayer.getDuration();
|
41 |
public
int getCurrentPosition()
throws RemoteException {
|
42 |
return
mPlayer.getCurrentPosition();
|
46 |
public
void seekTo( int
current) throws
RemoteException {
|
47 |
mPlayer.seekTo(current);
|
51 |
public
boolean setLoop( boolean
loop) throws
RemoteException {
|
58 |
public
void onCreate() {
|
59 |
Log.i(tag,
"MusicService onCreate()" );
|
60 |
mPlayer = MediaPlayer.create(getApplicationContext(), ElfPlayerUtil.getFileinSD( "wind.mp3" ));
|
64 |
public
IBinder onBind(Intent intent) {
|
7、其它代码和资源可以参见本讲附带的源代码,编译并运行程序,查看结果:
最后总结一下,AIDL提供了一种非常简单的方式,让我们可以把一个进程内的对象或方法暴露给另一个程序使用,就好象另一个程序也拥有这些功能一样。
最后感谢一首歌这个网站,本讲的图片素材采用的是他们的UI元素,好了,本讲就到这里。
分享到:
相关推荐
本例除了将音乐放在后台播放外,还实现了Activity中显示一个进度条,这个进度条要跟着Service中的MediaPlayer中的歌曲进度同步向前走,而且如果我点击进度条中的某一个位置,就让歌曲跳转到相应时间点继续播放。
两个工程导入一下,先运行service, 再运行client, 点击Button,就可以看到从Service中弹出的Toast, 内容是从Button那里发送过去的
Android调用远程服务(AIDL);两个APP,一个远程服务端APP,用于提供远程服务;一个客户端APP,用户绑定远程服务,并调用服务里面的方法
Android:AIDL和远程Service调用 本讲的内容,理解起来很难,也许你看了很多资料也看不明白,但是用起来缺简单的要命。所以我们干脆拿一个音乐播放器中进度条的实例来说明一下AIDL和Remote Service的价值和使用方法...
AIDLService,远程调用例子。包括服务端与客户端两个工程文件
AIDLService,远程调用例子。包括服务端与客户端两个工程文件源码,有兴趣的伙伴们抽时间可以看一下把。
AIDLService,远程调用例子。包括服务端与客户端两个工程文件.zip,太多无法一一验证是否可用,程序如果跑不起来需要自调,部分代码功能进行参考学习。
实现了android的远程service调用哪个,使用了AIDL,并且提供了一个自定义类型的参数
Activity与远程Service通信,使用AIDL
花了不少时间来研究,收个五分应该不过份吧,哈哈! 代码运行一定要先运行service应用,在运行cilent调用代码。
简单自定义aidl远程service调用实例
我们知道4个Android应用程序组件中的3个(Activity、BroadcastReceiver和ContentProvider)都可以进行跨进程访问,另外一个Android应用程序组件Service同样可以。因此,可以将这种可以跨进程访问的服务称为AIDL...
和博客文章http://blog.163.com/zhou_411424/blog/static/197362156201241541451/...对应改编的例子,实现了在一个APP1中通过AIDL远程调用另一个APP2的Service的函数,即使APP2没有被启动过! 均在eclipse编译运行通过。
Service的跨进程通信实战 设想这么一个场景,我们有2个APP(或者2个进程,均可),...PersonServer端是服务的提供者,我们首先需要创建一个Service来提供该服务,并且需要该服务具备跨进程通信的能力,以便Client端进行
简单远程service调用,aidl运用
我们知道4个Android应用程序组件中的3个(Activity、BroadcastReceiver和ContentProvider)都可以进行跨进程访问,另外一个Android应用程序组件Service同样可以。因此,可以将这种可以跨进程访问的服务称为AIDL...
第二十二讲:AIDL和远程Service调用 第二十三讲:Drawable使用入门 第二十四讲:Android动画入门(一) 第二十五讲:Android动画入门(二) 第二十六讲:Android中的GPS应用入门 第二十七讲:Handler使用...
Android系统采用了远程过程调用(RPC)方式来实现跨进程调用服务(Service),对于Service的跨进程调用需要通过AIDL来实现,关于如何实现aidl service请看本文介绍