Android-使用AIDL进程间通信
Android-使用AIDL进程间通信
发布时间:2015-06-05 来源:查字典编辑
摘要:对于客户端和服务之间通信,虽然可以通过Ibinder实现,但需要共享业务实现,如果在进程间通信的haunted,需要使用AIDL(Andro...

对于客户端和服务之间通信,虽然可以通过Ibinder实现,但需要共享业务实现,如果在进程间通信的haunted,需要使用AIDL(Android Interface Definition Language)进行。

AIDL是一种接口定义语言,用于约束两个进程间的通讯规则,编译器生成代码,实现Android设备上的两个进程间通信(IPC),AIDL的IPC机制和EJB所采用的CORBA很类似,进程之间的通信信息,首先会被转换成AIDL协议消息,然后发送给对方,对方收到AIDL协议消息后在转换成相应的对象。由于进程之间的通信信息需要双向转换,所以android采用代理类背后实现了信息的双向转换,代理类由android编译器生成,对开发人员来说是透明的。

使用方式如下:

1、 定义AIDL(同接口相似,但没有可见性,扩展名有.java—>.aidl)

//IdownloadService.aidl,注意扩展名

package cn.itcast.aidl;

interface IdownloadService{

void download(in/out/input String path);//in|out|inout是参数的方向。

}

Ide会自动在gen包下生成对应的java类,接口文件中生成一个stub的抽象类,里面包括aidl定义的方法,还包括一些其它辅助方法。值得关注的是asInterface(IBinder iBinder),它返回接口类型的实例,对于远程服务调用,远程服务返回给客户端的对象,客户端onServiceConnectionted(ComponentName name,IBinder service)方法引用该对象时不能直接强转成接口类型的实例,而应该使用asInterface(IBinder iBinder)进行类型转换。

编写AIDL需要注意:

1. 接口名和aidl文件相同。

2. 接口和方法前不用加访问权限修饰符public,private,protected等,也不能用final,static.

3. Aidl默认支持的类型包括java基本类型(int,long,boolean等)和(String,List,Map,CharSequence),使用这些类型是不需要import声明,对于List和Map中的元素类型必须是Aidl支持的类型,如果用自定义类型作为参数或返回值,自定义类型必须实现Parcelable接口。

4. 自定义类型的AIDL生成的其它接口类型在aidl描述文件中,应该显示import,即便在该类型和定义的包同一个包中。

5. 在aidl文件中所有非Java基本类型参数必须加上in、out、inout标记,以指明参数是输入参数,输出参数还是输入输出参数。

6.Java原始类型默认的标记位in,不能为其它标记。

Javabean必须实现Parcelable接口

class Person implements Parcelable{

id,name;

public int describeContents(){

return 0;

}

//把javabean中的数据写到Parcel

public void writeToParcel(Parcel dest,int flags){

dest.writeInt(this.id);

dest.writeString(this.name);

}

//添加一个静态成员,名为CREATOR,该对象实现了Parcelable.Creator接口

public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>(){

public Person createFromParcel(Parcel source) { return new Person(source.readInt(), source.readString());

}

public Person[] newArray(int size) {

return new Person[size];

}};

}

在自定义类型包中定义aidl声明文件

//Person.aidl,注意Parcelable是小写

Package cn.itcast.domain;

parcelable Person;

interface cn.itcast.domain.Person;

interface IPersonService{

void save(in Person person);

}

创建aidl接口实现类(通过继承${业务接口}.stub类实现)

Public class ServiceBinder extends IPersonService.Stub{

Public void save(Person person) throws RemoteException{

Log.i(“PersonService”,person.getId()+”=”+person.getName());

}

}

实现service的onBind方法,返回值就是上一步创建的aidl实现类对象。

public IBinder onBind(Intent intent){

return new ServiceBinder();

}

客户端通过隐式意图访问服务。

<service android:name=”.PersonService”>

<intent-filter>

<action android:name=”cn.itcast.process.aidl.PersonService”/>

new Intent(“cn.itcast.process.aidl.PersonService”);

复制aidl文件和所在包到客户端对应的src下。(客户端会自动生成对应java类)

this.bindService(,this.sc,BIND_AUTO_CREATE);

sc = new ServiceConnection(){

public void onServiceConnected(ComponentName,IBinder service){

personService = IPersonService.Stub.asInterface(service);

personService.save(new Person(56,”liming”));

}

public void onServiceDisconnected(ComponetName name){

personService = null;

}

}

Android-结束通话

Android没有对外公开通话的API,如果需要结束通话,必须使用AIDL于电话管理服务进行通行,并调用服务中的API实现结束通话,方法如下:

1. 从Android的源代码中拷贝以下文件到项目中:

com/android/internal/telephony/ITelephony.aidl

android/telephony/NeighboringCellInfo.aidl

如右图所示。开发工具会在gen目录下自动生成ITelephony.java

2. 调用ITelephony.endCall()结束通话:

Method method = Class.forName(“android.os.ServiceManager”)

.getMethod(“getService”,String.class);

IBinder binder = (IBinder)method.invoke(null,new Object[]{TELEPHONY_SERVICE});

ITelephony telephony = ITelephony.Stub.asInterface(binder);

Telephony.endCall();

在清单文件AndroidManifest.xml中添加权限

<uses-permission android:name=”android.permission.CALL_PHONE”/
>

推荐文章
猜你喜欢
附近的人在看
推荐阅读
拓展阅读
相关阅读
网友关注
最新安卓软件开发学习
热门安卓软件开发学习
编程开发子分类