对于客户端和服务之间通信,虽然可以通过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”/>