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

AIDL (Android Interface Definition Language)是一种IDL语言,用于生成可以在Android设备上两个进程之间进行进程间通信(IPC)的代码。如果在一个进程中(例如Activity)要调用另一个进程中(例如Service)对象的操作,就可以使用AIDL生成可序列化的参数。

AIDL IPC机制是面向接口的,它是使用代理类在客户端和实现端传递数据。

使用AIDL实现IPC

使用AIDL实现IPC服务的步骤是:

1.创建.aidl文件-该文件(YourInterface.aidl)定义了客户端可用的方法和数据的接口。

2.makefile文件中加入.aidl文件-Eclipse中的ADT插件提供管理功能)Android包括名为AIDL的编译器,位于tools/文件夹。

3.实现接口-AIDL编译器从AIDL接口文件中利用Java语言创建接口,该接口有一个继承的命名为Stub的内部抽象类(并且实现了一些IPC调用的附加方法),要做的就是创建一个继承于YourInterface.Stub的类并且实现在.aidl文件中声明的方法。

4.向客户端公开接口-如果是编写服务,应该继承Service并且重载Service.onBind(Intent)以返回实现了接口的对象实例

创建.aidl文件

AIDL使用简单的语法来声明接口,描述其方法以及方法的参数和返回值。这些参数和返回值可以是任何类型,甚至是其他AIDL生成的接口。重要的是必须导入所有非内置类型,哪怕是这些类型是在与接口相同的包中。下面是AIDL能支持的数据类型:

1.Java编程语言的主要类型(int, boolean)不需要import语句。

2.以下的类(不需要import语句):

String

List-列表中的所有元素必须是在此列出的类型,包括其他AIDL生成的接口和可打包类型。List可以像一般的类(例如List<String>)那样使用,另一边接收的具体类一般是一个ArrayList,这些方法会使用List接口。

Map- Map中的所有元素必须是在此列出的类型,包括其他AIDL生成的接口和可打包类型。一般的maps(例如Map<String,Integer>)不被支持,另一边接收的具体类一般是一个HashMap,这些方法会使用Map接口。

CharSequence-该类是被TextView和其他控件对象使用的字符序列。

3.通常引引用方式传递的其他AIDL生成的接口,必须要import语句声明

4.实现了Parcelable protocol以及按值传递的自定义类,必须要import语句声明。

通过对上面的基本了解,下面我就以一个具体的实例来说明Android中如何通过AIDL机制来实现两个进程中实现通讯:(情景假设:例如A应用通过服务Service方式向B应用提供通过书籍编号来查询书籍名称的服务)

A应用程序结构图如下:

Android应用中通过AIDL机制实现进程间的通讯实例

通过上面的结构图可以看到,在A应用程序中创建一个 aidl 的接口,然后系统在 gen 目录下自动生成相应的 java 文件。

其中 IBook.aidl 文件的源码:

[html] view plain copy
  1. packagecom.andyidea.aidl;
  2. interfaceIBook{
  3. StringqueryBook(intbookNo);
  4. }

其中 BookService.java 类中的源码如下:

[html] view plain copy
  1. packagecom.andyidea.service;
  2. importcom.andyidea.aidl.IBook;
  3. importandroid.app.Service;
  4. importandroid.content.Intent;
  5. importandroid.os.IBinder;
  6. importandroid.os.RemoteException;
  7. /**
  8. *查询书籍的服务
  9. *@authorAndy
  10. *
  11. */
  12. publicclassBookServiceextendsService{
  13. privateString[]bookNames={"Java编程思想","设计模式","Android开发设计"};
  14. privateIBindermIBinder=newBookBinder();
  15. @Override
  16. publicIBinderonBind(Intentintent){
  17. //TODOAuto-generatedmethodstub
  18. returnmIBinder;
  19. }
  20. /**
  21. *服务中交互的方法
  22. *@parambookNo
  23. *@return
  24. */
  25. publicStringqueryBookName(intbookNo){
  26. if(bookNo>0&&bookNo<=bookNames.length){
  27. returnbookNames[bookNo-1];
  28. }
  29. returnnull;
  30. }
  31. privateclassBookBinderextendsIBook.Stub{
  32. @Override
  33. publicStringqueryBook(intbookNo)throwsRemoteException{
  34. returnqueryBookName(bookNo);
  35. }
  36. }
  37. }
同时别忘了在 Manifest.xml中配置该服务对象(标红色的部分),建议采用隐式方式激活该服务,适合不同的进程的意图。

[html] view plain copy
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <manifestxmlns:android="http://schemas.android.com/apk/res/android"
  3. package="com.andyidea.service"
  4. android:versionCode="1"
  5. android:versionName="1.0">
  6. <uses-sdkandroid:minSdkVersion="8"/>
  7. <application
  8. android:icon="@drawable/ic_launcher"
  9. android:label="@string/app_name">
  10. <spanstyle="color:#ff0000;"><serviceandroid:name=".BookService">
  11. <intent-filter>
  12. <actionandroid:name="com.andyidea.aidl.bookservice"/>
  13. </intent-filter>
  14. </service></span>
  15. </application>
  16. </manifest>
以上我们已经实现了A应用程序提供服务的功能,下面我们来实现B应用(或者其它需要用到A应用提供服务的应用程序)

B应用程序结构图如下:

Android应用中通过AIDL机制实现进程间的通讯实例

我们看到B应用程序也要和服务端同样的 .aidl 文件,我们可以从A应用程序中把该 aidl 文件中拷贝过来就是了,呵。由于B应用中 .aidl 文件和 A应用中的 .aidl 文件源码一样,我在这里就不列出来了。

其中AIDLClientDemoActivity.java源码如下:【注:其中该客户端类要通过 bindService 方式来启动另外一个进程的服务,这样才能实现和服务进行交互。如果通过startService方式来启动服务,则不能与服务进行交互】

[html] view plain copy
  1. packagecom.andyidea.client;
  2. importandroid.app.Activity;
  3. importandroid.content.ComponentName;
  4. importandroid.content.Intent;
  5. importandroid.content.ServiceConnection;
  6. importandroid.os.Bundle;
  7. importandroid.os.IBinder;
  8. importandroid.os.RemoteException;
  9. importandroid.view.View;
  10. importandroid.widget.Button;
  11. importandroid.widget.EditText;
  12. importandroid.widget.TextView;
  13. importcom.andyidea.aidl.IBook;
  14. publicclassAIDLClientDemoActivityextendsActivity{
  15. privateEditTextnumberText;
  16. privateTextViewresultView;
  17. privateButtonquery;
  18. privateIBookbookQuery;
  19. privateBookConnectionbookConn=newBookConnection();
  20. /**Calledwhentheactivityisfirstcreated.*/
  21. @Override
  22. publicvoidonCreate(BundlesavedInstanceState){
  23. super.onCreate(savedInstanceState);
  24. setContentView(R.layout.main);
  25. numberText=(EditText)this.findViewById(R.id.number);
  26. resultView=(TextView)this.findViewById(R.id.resultView);
  27. query=(Button)findViewById(R.id.query);
  28. Intentservice=newIntent("com.andyidea.aidl.bookservice");
  29. bindService(service,bookConn,BIND_AUTO_CREATE);
  30. query.setOnClickListener(newView.OnClickListener(){
  31. @Override
  32. publicvoidonClick(Viewv){
  33. Stringnumber=numberText.getText().toString();
  34. intnum=Integer.valueOf(number);
  35. try{
  36. resultView.setText(bookQuery.queryBook(num));
  37. }catch(RemoteExceptione){
  38. e.printStackTrace();
  39. }
  40. }
  41. });
  42. }
  43. @Override
  44. protectedvoidonDestroy(){
  45. unbindService(bookConn);
  46. super.onDestroy();
  47. }
  48. privatefinalclassBookConnectionimplementsServiceConnection{
  49. @Override
  50. publicvoidonServiceConnected(ComponentNamename,IBinderservice){
  51. //TODOAuto-generatedmethodstub
  52. bookQuery=IBook.Stub.asInterface(service);
  53. }
  54. @Override
  55. publicvoidonServiceDisconnected(ComponentNamename){
  56. //TODOAuto-generatedmethodstub
  57. bookQuery=null;
  58. }
  59. }
  60. }
其中界面布局文件 main.xml 源码:

[html] view plain copy
  1. <?xmlversion="1.0"encoding="utf-8"?>
  2. <LinearLayoutxmlns:android="http://schemas.android.com/apk/res/android"
  3. android:layout_width="fill_parent"
  4. android:layout_height="fill_parent"
  5. android:orientation="vertical">
  6. <TextView
  7. android:layout_width="fill_parent"
  8. android:layout_height="wrap_content"
  9. android:text="书籍编号"/>
  10. <EditText
  11. android:id="@+id/number"
  12. android:layout_width="fill_parent"
  13. android:layout_height="wrap_content"/>
  14. <Button
  15. android:id="@+id/query"
  16. android:layout_width="wrap_content"
  17. android:layout_height="wrap_content"
  18. android:text="查询"/>
  19. <TextView
  20. android:id="@+id/resultView"
  21. android:layout_width="fill_parent"
  22. android:layout_height="wrap_content"/>
  23. </LinearLayout>
至此,我们已经完成了B应用程序的代码实现部分,我们要先把A应用程序部署到机器上,然后我们再运行B应用程序。下面我们通过截图来看下程序运行的结果:
Android应用中通过AIDL机制实现进程间的通讯实例

通过上面的截图,我们输入书籍编号 1,就可以查询出相应的书籍名称,到此,我们就可以了解了 Android应用中如何通过AIDL机制实现两个进程的通讯。

注:本文为 Andy.Chen 原创,欢迎大家转载,转载请大家注明出处,谢谢。

更多相关文章

  1. Android studio 下的aidl编程实现Android的夸进程间通信
  2. Android 4.1模拟器访问服务器接口
  3. Android多线程系统概述(sundy深入浅出)之进程和线程
  4. android进程间通讯(3)--使用socket
  5. Android进程说明

随机推荐

  1. android 开发效率
  2. recovery 根据@/cache/recovery/block.ma
  3. 是广告还是病毒
  4. [置顶] AIDL旅行记之AIDL的简单使用
  5. 如何通过安卓来为自己赚钱
  6. 广播+ Service详解、通知和权限-Android
  7. Android(安卓)Develop Training——和其
  8. comScore:Android作業系統在美國市佔率高
  9. Android(安卓)Trick 6: PendingIntent的
  10. 超好用的Android日志打印框架--Logger