当前位置:操作系统 > 安卓/Android >>

Android Service和Binder、AIDL

1.首先理解service的作用和生命周期,
 
由于activity如果切换,那么他就不再运行,那么我们想在玩游戏的时候听播放器中的音乐,activity就应运而生了,这是最常见的一种场景,同时service由于它的优先级比较高,不容易被回收,而且是独立进程,不会阻塞UI线程,因此,可以用来处理一些比较费时的任务。
 
service起于startService(),终于stopService,如果没有调用stopService,那么,即使调用者结束了,该service也一直存在。
 
也可以通过bindService来绑定service,unBindService分开并结束service。如果bind的时候没有启动service,那么它会调用service的create方法启动。
 
多个程序可同时bind同一个service,只有他们都unbind了,这个service就会自动结束。如果有部分unbinder的时候调用stopservice,这个stop也会等到它们全部结束的时候才真的结束。
 
 
 
2.service的种类,
 
由于adroid的独特的线程模型,service被同一个apk调用和不同apk调用原理是不同的,因此分成以下两种:
 
(1). 本地服务(Local Service):说白了就是在同一个apk内被调用。
 
(2). 远程服务(Remote Sercie):被另外一个apk调用。
 
 
 
3.涉及到的内容:
 
由于android的进程模型,不同的apk不能共用数据,因此如果有需要的话需要通过进程间通讯完成。
 
进程间通讯(ipc)涉及到三个部分:客户端(调用方),传递数据,服务端(被调用方)。
 
android的数据传递类似于RPC过程,采用aidl的方式传递,考虑到效率的原因,没有用java内置的serializable,而是采用原始数据拆分组装的parcel方式。
 
 
 
4.场景假设:
 
一个service提供产生Student数据的功能,作为服务端,一个apk中的activity想获取另一个apk中的一个学生。根据上边的分析,service即使服务端,activity即使客户端,student是要传输的数据,要被包装成pacel传递。
 
 
 
5.根据android思想,理想的传递过程:
 
student提供拆分成parcel的writeToParcel()方法和根据Parcel组装的构造函数Student(Pacel p);
 
一个bissiness Service(bizSvc)用来实现逻辑功能。
 
一个android service(adSvc)调用bizService提供功能。
 
一个MyBinder负责service描述符与bizService的映射(本地服务时供queryLocalInterface查找)和与客户端交互。(如果被本地调用,就查找到相应的service直接操作,不用远程通讯,如果被远程调用,就得负责与远程通讯)
 
一个Proxy负责在客户端提供transact()方法发送数据。
 
 
 
具体的调用过程如下:
 
调用的时候,客户端首先调用bindService(new Intent ("abc"), serviceConnection,Context.BIND_AUTO_CREATE);激活serviceConnection的onServiceConnected方法,在此方法中获取到一个binder(注:clientBinder,系统给我们的一个与远程进行通信的binder,不是我们刚才自己实现的),此binder能够查找系统中注册的service,如果没有查找到该Service,那么可认定该service是从其他apk获得的,就创建一个此service的静态代理类Proxy,否则,就把这个service返回,供客户端调用。
 
 
 
服务端收到这个Intent后,激活adSvc的onBind方法,创建一个MyBinder返回。之后,这个MyBinder负责与客户端的Proxy通信。
 
 
 
之后,客户端要调用service的方法,可直接调用(本地)或者通过代理调用(远程),这个Proxy调用clientBinder的transact方法,参数为要掉用的方法(TRANSACRION_XX),发送的参数(_data),接受的返回值(_reply),把消息传送给服务端。
 
服务端收到消息后,调用MyBinder的onTransac方法,根据Proxy传递过来参数,调用bizService不同的方法,并把产生的值组装成Parcel发送回去。之后客户端Proxy会自动调用sudent的相关方法,把数据重新组装,进行下一步处理。
 
 
 
理想的代码如下(这是理想的代码,经过拆分,但是不符合android的aidl规范,不能运行,见下文分析):
 
客户端代码:
 
final IMyBizService bizSvc;
 
bindService(new Intent("abc"),new ServiceConnection(){
 
@Override
 
public void onServiceConnected(ComponentName name, IBinder clientBinder) {
 
if ((binder == null)) {
 
return null;
 
}
 
IInterface iin =  clientBinder.queryLocalInterface(DESCRIPTOR);//如果是查找本地有,就直接返回
 
if (((iin != null) && (iin instanceof IMyService))) {
 
return ((IMyService) iin);
 
}
 
return new Proxy(obj);//否则采用远程代理
 
 
 
}
 
}, Context.BIND_AUTO_CREATE);
 
 
 
Proxy.java:
 
public class Proxy implements IMyBizService {
 
private IBinder mRemote;
 
private String description;
 
 
 
public Proxy(IBinder binder,String description) {
 
super();
 
this.mRemote=binder;
 
this.description=description;
 
}
 
 
 
@Override
 
public IBinder asBinder() {
 
return mRemote;
 
}
 
 
 
public Student getStudent() throws android.os.RemoteException {
 
android.os.Parcel _data = android.os.Parcel.obtain();
 
android.os.Parcel _reply = android.os.Parcel.obtain();
 
Student _result;
 
try {
 
_data.writeInterfaceToken(description);
 
mRemote.transact(MyBinder.TRANSACTION_getStudent, _data,_reply, 0);//发送参数给服务端
 
_reply.readException();
 
if ((0 != _reply.readInt())) {
 
_result = new Student(_reply);//接受参数并返回
 
} else {
 
_result = null;
 
}
 
} finally {
 
_reply.recycle();
 
_data.recycle();
 
}
 
return _result;
 
}
 
 
 
}
 
 
 
 
 
服务端代码:
 
MyService.java:
 
public class MyService extends Service {
 
@Override
 
public IBinder onBind(Intent intent) {
 
MyBizServiceImpl svc=new MyBizServiceImpl();
 
MyBinder binder= new MyBinder(new MyBizServiceImpl(),intent.getAction());
 
svc.setBinder(binder);//(此处有点绕,binder方法需要bizService作为参数实现绑定和调用,而bizSvc需要这个Binder作为参数,用来实现asBind方法,形成了一个环);
 
return binder;
 
}
 
}
 
 
 
MyBinder.java:
 
public class MyBinder extends Binder {
 
static final int TRANSACTION_getStudent = (IBinder.FIRST_CALL_TRANSACTION + 1);
 
private IInterface intf;
 
private String descripter;
 
 
 
public MyBinder(IInterface intf, String descripter) {
 
super();
 
this.intf=intf;
 
this.descripter=descripter;
 
this.attachInterface(intf, descripter);
 
}
 
 
 
@Override
 
public boolean onTransact(int code, Parcel data,Parcel reply, int flags)throws RemoteException {
 
IMyBizService myService=(IMyBizService) this.intf;
 
switch (code) {
 
case INTERFACE_TRANSACTION: {
 
reply.writeString(descripter);
 
return true;
 
}
 
case TRANSACTION_getStudent: {
 
data.enforceInterface(descripter);
 
Student _result = myService.getStudent();
 
reply.writeNo
补充:移动开发 , Android ,
CopyRight © 2012 站长网 编程知识问答 www.zzzyk.com All Rights Reserved
部份技术文章来自网络,