文档章节

什么是aidl?Android AIDL详解

今幕明
 今幕明
发布于 2014/03/04 10:16
字数 2303
阅读 3205
收藏 6

什么是aidl:
aidl是 Android Interface definition language的缩写,一看就明白,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口
icp:interprocess communication :内部进程通信。

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

(1)在Eclipse Android工程的Java包目录中建立一个扩展名为aidl的文件。该文件的语法类似于Java代码,但会稍有不同。
(2)如果aidl文件的内容是正确的,ADT会自动生成一个Java接口文件(*.java)。
(3)建立一个服务类(Service的子类)。
(4)实现由aidl文件生成的Java接口。
(5)在AndroidManifest.xml文件中配置AIDL服务,尤其要注意的是,<action>标签中android:name的属性值就是客户端要引用该服务的ID,也就是Intent类的参数值。 

AIDL 是一种接口定义语言,用于约束两个进程间的通讯规则,供编译器生成代码,实现Android设备上的两个进程间通信(IPC)。AIDL的IPC机制和 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;

interface IDownloadService {

   void download(String path);

}

当 完成aidl文件创建后,eclipse会自动在项目的gen目录中同步生成IDownloadService.java接口文件。接口文件中生成一个 Stub的抽象类,里面包括aidl定义的方法,还包括一些其它辅助方法。值得关注的是asInterface(IBinder iBinder),它返回接口类型的实例,对于远程服务调用,远程服务返回给客户端的对象为代理对象,客户端在 onServiceConnected(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,不能为其它标记。

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

public class ServiceBinder extends IDownloadService.Stub {

   @Override

   public void download(String path) throws RemoteException {

      Log.i("DownloadService", path);

   }    

}

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

public class DownloadService extends Service {

   private ServiceBinder serviceBinder = new ServiceBinder();

   @Override

   public IBinder onBind(Intent intent) {

      return serviceBinder;

   }

   public class ServiceBinder extends IDownloadService.Stub {

      @Override

      public void download(String path) throws RemoteException {

        Log.i("DownloadService", path);

      }    

   }

}

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

<service android: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应用通信,代码如下:

public class ClientActivity extends Activity {

   private IDownloadService downloadService;

   @Override

   public void onCreate(Bundle savedInstanceState) {

      super.onCreate(savedInstanceState);

      setContentView(R.layout.main);

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

   }

   @Override

   protected void onDestroy() {

      super.onDestroy();

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

   } 

   private ServiceConnection serviceConnection = 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基本类型(int、long、boolean等)和(String、List、Map、CharSequence),如果要传递自定义的类型该如何实现呢?

进程间传递自定义类型的实现过程请参见页面下方备注栏:

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

   private Integer id;

   private String name;

   public Person(){}

   public Person(Integer id, String name) {

      this.id = id;

      this.name = name;

   }

   public Integer getId() {

      return id;

   }

   public void setId(Integer id) {

      this.id = id;

   }

   public String getName() {

      return name;

   }

   public void setName(String name) {

      this.name = name;

   } 

   @Override

   public int describeContents() {

      return 0;

   }

   @Override

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

      dest.writeInt(this.id);

      dest.writeString(this.name);

   }

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

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

      @Override

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

        return new Person(source.readInt(), source.readString());

      }

      @Override

      public Person[] newArray(int size) {

        return new Person[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(in Person person);

}

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

public class ServiceBinder extends IPersonService.Stub {

       @Override

       public void save(Person person) throws RemoteException {

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

       }   

}

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

public class PersonService extends Service {

   private ServiceBinder serviceBinder = new ServiceBinder();

   @Override

   public IBinder onBind(Intent intent) {

      return serviceBinder;

   }

public class ServiceBinder extends IPersonService.Stub {

       @Override

       public void save(Person person) throws RemoteException {

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

       }

}

}

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

<service android:name=".PersonService" >

   <intent-filter>

      <action android: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 {

   private IPersonService personService;

   @Override

   public void onCreate(Bundle savedInstanceState) {

      super.onCreate(savedInstanceState);

      setContentView(R.layout.main);

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

   }

@Override

   protected void onDestroy() {

      super.onDestroy();

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

   } 

   private ServiceConnection serviceConnection = new ServiceConnection() {

      @Override

      public void onServiceConnected(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 void onServiceDisconnected(ComponentName name) {

        personService = null;

      }

   };

}

最后我们在总结一下android中 本地服务和 AIDL服务的区别:
    本地服务不支持onBind(),它从onBind()返回null,这种类型的服务只能由承载服务的应用程序组件访问。可以调用 startService()来调用本地服务。AIDL服务可以同时供 同一进程内的组件和其他应用程序的组件使用。这种类型的服务在AIDL 文件中为自身与其客户端定义一个契约。服务实现 AIDL契约,而客户端绑定到 AIDL定义。服务通过从 onBind()方法 返回AIDL接口的实现,来实现契约。客户端通过调用 bindService()来绑定到AIDL服务,并调用 unBindService()来从服务断开。


本文转载自:http://www.cr173.com/html/19652_1.html

今幕明
粉丝 47
博文 230
码字总数 39350
作品 0
朝阳
程序员
私信 提问
aidl详解续

上次撤了一些但之后,这次来详细说说aidl到底是个什么玩意。现在在网上有好多关于aidl的资料但是国内大部分呢还是都sopy的所以基本上看了一个就没有必要看其他的了,其中有一个说aidl是主要用...

找到组织
2013/12/19
751
3
安卓跨进程通信之Aidl教程详解(一)

安卓多进程通信主要有四大方案:1,Activity 2,Broadcast Receiver 3,Content Prodiver 4,Service 1,Activity Intent intent = new Intent(); intent.setAction(Intent.ACTION_CALL); inte......

修炼爱情
2017/04/19
0
0
Android AIDL使用详解

1.什么是aidl:aidl是 Android Interface definition language的缩写,一看就明白,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口 icp:interprocess commu...

gavin_jin
2012/03/18
455
0
Android中aidl如何import文件

如果你做Android开发,那就应该会知道aidl工具的,但是会使用aidl命令行工具的人有吗有吗有吗,如果有的话你们为神马不出来写写怎么用!!!! 用aidl命令行的如果不出意外肯定会遇见类似这样...

华宰
2011/09/08
4.1K
0
Android高手进阶教程(二十五)之---Android 中的AIDL!!!

大家好,好久不见,今天要给大家分享的是android aidl的使用。在Android中, 每个应用程序都可以有自己的进程. 在写UI应用的时候, 经常要用到Service. 在不同的进程中, 怎样传递对象呢? 显然,...

迷途d书童
2012/02/29
304
1

没有更多内容

加载失败,请刷新页面

加载更多

安装k8s集群(一):window10下安装三台centOS

第一步:开启window10的Hyper-V 1. Hyper-V简介 Hyper-V是window提供的虚拟化产品,是window版本的vmware,window 10操作系统(专业版和企业版)默认集成了hyper-V。 体积小、运行速度快、轻...

TopDuang
39分钟前
4
0
mac环境安装TunnelBrick失败

mac最新环境安装TunnelBrick失败。 os系统版本 10.15 解决办法: 参考链接 (1)安装tuntap brew cask install tuntap (2)执行以下命令 kextstat | grep tapsudo kextunload -b net.sf.tu...

wotrd
42分钟前
4
0
java线程池,Executor框架,Callable,Future

package com.yh.tools.thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory;import java.util.ArrayList;import java.util.List;import java.util.concurrent.Cal......

颖辉小居
43分钟前
4
0
Python3.8正式发布,这些特性令人眼前一亮

在2019年10月14号,python38第一版以正式发布。所以屏幕面前的各位2.7忠粉,你们到底还在等什么❓那么3.8版本能给大家带来怎样的全新体验,让我们一起看看吧。 > 🍺本篇文章大约需要2分钟来...

s045pd
50分钟前
4
0
Git Reset 三种模式hard,soft,mix

有时候,我们用Git的时候有可能commit提交代码后,发现这一次commit的内容是有错误的,那么有两种处理方法: 1、修改错误内容,再次commit一次 2、使用git reset 命令撤销这一次错误的commi...

筱飞
52分钟前
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部