Android:Service:AIDL实现进程通信

77 篇文章 0 订阅

Android, 每个应用程序都有自己的进程,当需要在不同的进程之间传递对象时,该如何实现呢? 显然, Java中是不支持跨进程内存共享的。因此要传递对象, 需要把对象解析成操作系统能够理解的数据格式, 以达到跨界对象访问的目的。在JavaEE中,采用RMI通过序列化传递对象。在Android, 则采用AIDL(Android Interface Definition Language:接口描述语言)方式实现。

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

实现进程通信,一般需要下面四个步骤:

假设A应用需要与B应用进行通信,调用B应用中的download(String path)方法,B应用以Service方式向A应用提供服务。需要下面四个步骤:

1> B应用中创建*.aidl文件,aidl文件的定义和接口的定义很相类,如:在cn.itcast.aidl包下创建IDownloadService.aidl文件,内容如下:

package cn.itcast.aidl;

interfaceIDownloadService {

  void download(String path);

}

当完成aidl文件创建后,eclipse会自动在项目的gen目录中同步生成IDownloadService.java接口文件。接口文件中生成一个Stub的抽象类,里面包括aidl定义的方法,还包括一些其它辅助方法。值得关注的是asInterface(IBinderiBinder),它返回接口类型的实例,对于远程服务调用,远程服务返回给客户端的对象为代理对象,客户端在onServiceConnected(ComponentName name, IBinder service)方法引用该对象时不能直接强转成接口类型的实例,而应该使用asInterface(IBinderiBinder)进行类型转换。

编写Aidl文件时,需要注意下面几点:

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

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

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

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

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

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

2> B应用中实现aidl文件生成的接口(本例是IDownloadService),但并非直接实现接口,而是通过继承接口的Stub来实现(Stub抽象类内部实现了aidl接口),并且实现接口方法的代码。内容如下:

publicclass ServiceBinder extends IDownloadService.Stub {

  @Override

  public void download(String path) throws RemoteException {

  Log.i("DownloadService",path);

 

}

3> B应用中创建一个Service(服务),在服务的onBind(Intent intent)方法中返回实现了aidl接口的对象(本例是ServiceBinder)。内容如下:

publicclass DownloadService extendsService {

  private ServiceBinderserviceBinder = new ServiceBinder();

  @Override

  public IBinderonBind(Intent intent) {

  return serviceBinder;

  }

  public class ServiceBinder extends IDownloadService.Stub {

  @Override

  public void download(String path) throws RemoteException {

  Log.i("DownloadService",path);

 

  }

}

其他应用可以通过隐式意图访问服务,意图的动作可以自定义,AndroidManifest.xml配置代码如下:

<serviceandroid:name=".DownloadService">

  <intent-filter>

  <action android:name="cn.itcast.process.aidl.DownloadService"/>

  </intent-filter>

</service>

4> B应用中aidl文件所在package连同aidl文件一起拷贝到客户端A应用,eclipse会自动在A应用的gen目录中为aidl文件同步生成IDownloadService.java接口文件,接下来就可以在A应用中实现与B应用通信,代码如下:

publicclass ClientActivity extendsActivity {

  private IDownloadServicedownloadService;

  @Override

  public void onCreate(Bundle savedInstanceState) {

  super.onCreate(savedInstanceState);

  setContentView(R.layout.main);

  this.bindService(newIntent("cn.itcast.process.aidl.DownloadService"), this.serviceConnection,BIND_AUTO_CREATE);//绑定到服务

  }

  @Override

  protected void onDestroy() {

  super.onDestroy();

  this.unbindService(serviceConnection);//解除服务

 

 

  private ServiceConnectionserviceConnection = new ServiceConnection() {

  @Override

  public void onServiceConnected(ComponentName name, IBinder service){

  downloadService = IDownloadService.Stub.asInterface(service);

  try {

  downloadService.download("http://www.itcast.cn");

  } catch (RemoteException e) {

  Log.e("ClientActivity", e.toString());

  }

  }

  @Override

  public void onServiceDisconnected(ComponentName name) {

  downloadService = null;

  }

  };

}


进程间传递自定义类型参数

Aidl默认支持的类型包话java基本类型(intlongboolean等)和(StringListMapCharSequence),如果要传递自定义的类型该如何实现呢?

要传递自定义类型,首先要让自定义类型支持parcelable协议,实现步骤如下:

1>自定义类型必须实现Parcelable接口,并且实现Parcelable接口的publicvoid writeToParcel(Parcel dest, int flags)方法 。

2>自定义类型中必须含有一个名称为CREATOR的静态成员,该成员对象要求实现Parcelable.Creator接口及其方法。

3> 创建一个aidl文件声明你的自定义类型。

Parcelable接口的作用:实现了Parcelable接口的实例可以将自身的状态信息(状态信息通常指的是各成员变量的值)写入Parcel,也可以从Parcel中恢复其状态。Parcel用来完成数据的序列化传递。

1> 创建自定义类型,并实现Parcelable接口,使其支持parcelable协议。如:在cn.itcast.domain包下创建Person.java:

package cn.itcast.domain;

import android.os.Parcel;

import android.os.Parcelable;

public class Person implements Parcelable

  privateInteger id;

  private Stringname;

 

  public Person(){}

  publicPerson(Integer id, String name) {

  this.id = id;

  this.name = name;

  }

  public IntegergetId() {

  return id;

  }

  public voidsetId(Integer id) {

  this.id = id;

  }

  public StringgetName() {

  return name;

  }

  public voidsetName(String name) {

  this.name = name;

 

  @Override

  public intdescribeContents() {

  return 0;

  }

  @Override

  public voidwriteToParcel(Parcel dest, int flags) {//javanbean中的数据写到Parcel

  dest.writeInt(this.id);

  dest.writeString(this.name);

  }

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

  publicstatic final Parcelable.Creator<Person> CREATOR = newParcelable.Creator<Person>(){

  @Override

  public PersoncreateFromParcel(Parcel source) {//Parcel中读取数据,返回person对象

  return newPerson(source.readInt(), source.readString());

  }

  @Override

  public Person[]newArray(int size) {

  return newPerson[size];

  }

  };

}

2> 在自定义类型所在包下创建一个aidl文件对自定义类型进行声明,文件的名称与自定义类型同名。

package cn.itcast.domain;

parcelable Person;

3> 在接口aidl文件中使用自定义类型,需要使用import显式导入,本例在cn.itcast.aidl包下创建IPersonService.aidl文件,内容如下:

package cn.itcast.aidl;

import cn.itcast.domain.Person;

interface IPersonService {

      void save(inPerson person);

}

4> 在实现aidl文件生成的接口(本例是IPersonService),但并非直接实现接口,而是通过继承接口的Stub来实现(Stub抽象类内部实现了aidl接口),并且实现接口方法的代码。内容如下:

public class ServiceBinder extends IPersonService.Stub {

       @Override

       public voidsave(Person person) throws RemoteException {

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

      

}

5> 创建一个Service(服务),在服务的onBind(Intent intent)方法中返回实现了aidl接口的对象(本例是ServiceBinder)。内容如下:

public class PersonService extends Service {

  privateServiceBinder serviceBinder = new ServiceBinder();

  @Override

  public IBinderonBind(Intent intent) {

  returnserviceBinder;

  }

publicclass ServiceBinder extends IPersonService.Stub {

       @Override

       public void save(Person person) throwsRemoteException {

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

       }

}

}

其他应用可以通过隐式意图访问服务,意图的动作可以自定义,AndroidManifest.xml配置代码如下:

<service android:name=".PersonService" >

  <intent-filter>

  <actionandroid:name="cn.itcast.process.aidl.PersonService " />

  </intent-filter>

</service>

6> 把应用中的aidl文件和所在package一起拷贝到客户端应用的src目录下,eclipse会自动在客户端应用的gen目录中为aidl文件同步生成IPersonService.java接口文件,接下来再把自定义类型文件和类型声明aidl文件及所在package一起拷贝到客户端应用的src目录下。

最后就可以在客户端应用中实现与远程服务的通信,代码如下:

public class ClientActivity extends Activity {

  privateIPersonService personService;

  @Override

  public voidonCreate(Bundle savedInstanceState) {

  super.onCreate(savedInstanceState);

  setContentView(R.layout.main);

  this.bindService(newIntent("cn.itcast.process.aidl.PersonService"),this.serviceConnection, BIND_AUTO_CREATE);//绑定到服务

  }

  @Override

  protected voidonDestroy() {

  super.onDestroy();

  this.unbindService(serviceConnection);//解除服务

 

 

  privateServiceConnection serviceConnection = new ServiceConnection() {

  @Override

  public voidonServiceConnected(ComponentName name, IBinder service) {

  personService =IPersonService.Stub.asInterface(service);

  try {

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

  } catch(RemoteException e) {

  Log.e("ClientActivity",e.toString());

  }

  }

  @Override

  public voidonServiceDisconnected(ComponentName name) {

  personService =null;

  }

  };

}


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

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值