Android AIDL的用法

一、什么是AIDL服务

  一般创建的服务并不能被其他的应用程序访问。为了使其他的应用程序也可以访问本应用程序提供的服务,Android系统采用了远程过程调用(Remote Procedure Call,RPC)方式来实现。与很多其他的基于RPC的解决方案一样,Android使用一种接口定义语言(Interface Definition Language,IDL)来公开服务的接口。因此,可以将这种可以跨进程访问的服务称为AIDL(Android Interface Definition Language)服务。

二、AIDL基本语法

  AIDL使用简单的语法来声明接口,描述其方法以及方法的参数和返回值。这些参数和返回值可以是任何类型,甚至是其他AIDL生成的接口。
  其中对于Java编程语言的基本数据类型 (int, long, char, boolean,String,CharSequence)集合接口类型List和Map,不需要import 语句。
  而如果需要在AIDL中使用其他AIDL接口类型,需要import,即使是在相同包结构下。AIDL允许传递实现Parcelable接口的类,需要import
  需要特别注意的是,对于非基本数据类型,也不是String和CharSequence类型的,需要有方向指示,包括in、out和inout,in表示由客户端设置,out表示由服务端设置,inout是两者均可设置。

  AIDL只支持接口方法,不能公开static变量。

三、android应用层使用AIDL

3.1、简要步骤

  1、在Eclipse Android工程的Java包目录中建立一个扩展名为A.aidl的文件,并写下需要的接口。如果aidl文件的内容是正确的,ADT会在gen目录下自动生成一个A.Java接口文件。
  2、建立一个服务类(Service的子类)。并在创建的服务类中创建一个内部类,实现由aidl文件生成的Java接口。
  3、在服务类的onBind方法返回时,将实现aidl接口的内部类对象返回出去。
  4、在AndroidManifest.xml文件中配置AIDL服务,尤其要注意的是,<action>标签中android:name的属性值就是客户端要引用该服务的ID,也就是Intent类的参数值。

3.2、具体操作

3.2.1、创建文件IMyService.aidl:


  文件内容:

  1. package du.pack;
  2. interface IMyService{
  3. //只有一个接口
  4. String getValue();
  5. }

3.2.2、创建服务类及实现内部类

  1. public class MyService extends Service {
  2. @Override
  3. public IBinder onBind(Intent arg0) {
  4. // 把内部类的对象返回给客户端使用
  5. return new MyServiceImpl();
  6. }
  7. // 创建一个继承自IMyService.Stub的内部类
  8. public class MyServiceImpl extends IMyService.Stub {
  9. // 必须实现AIDL文件中的接口
  10. public String getValue() throws RemoteException {
  11. return null;
  12. }
  13. }
  14. }

  注意,我们写的service中,onBind方法必须返回MyServiceImpl类的对象实例,否则客户端无法获得服务对象。

3.2.3、在AndroidManifest.xml文件中配置MyService类

  1. <service android:name=".MyService" >
  2. <intent-filter>
  3. <action android:name="du.pack.IMyService" />
  4. </intent-filter>
  5. </service>

  上面的"du.pack.IMyService"是客户端用于访问AIDL服务的ID。

4、本地客户端的使用方法

4.1、新建一个Eclipse Android工程,并将刚才远程服务端自动生成的gen目录下面的IMyService.java文件连同包目录一起复制到客户端工程的src目录中。

4.2、调用AIDL服务首先要绑定服务,然后才能获得服务对象。

  1. public class AidlClientTestActivity extends Activity {
  2. // 远程服务端的对象
  3. IMyService mIMyService;
  4. private ServiceConnection mConnection = new ServiceConnection() {
  5. public void onServiceConnected(ComponentName name, IBinder service) {
  6. // 绑定成功,得到远程服务端的对象,目标完成!!!
  7. mIMyService = IMyService.Stub.asInterface(service);
  8. }
  9. public void onServiceDisconnected(ComponentName name) {
  10. // 解除绑定
  11. mIMyService = null;
  12. }
  13. };
  14. @Override
  15. public void onCreate(Bundle savedInstanceState) {
  16. super.onCreate(savedInstanceState);
  17. setContentView(R.layout.main);
  18. // 绑定远程服务端服务
  19. Intent serviceIntent = new Intent("du.pack.IMyService");
  20. bindService(serviceIntent, mConnection, Context.BIND_AUTO_CREATE);
  21. }
  22. }

5、用法小结

  回顾一下整个调用的过程:
  服务端方面:将需要开放的接口抽象到aidl文件中,然后在自己的内部类中对接口进行实现,并在自己被绑定的时候把内部类对象返回给客户端。客户端方面:当我们需要远程某个Service时,就像在绑定本地的Service一样去bindService,然后在绑定成功的回调函数中(也就是onServiceConnected方法)得到一个Ibinder对象(比如Service),这时调用IMyService.Stub.asInterface(service)这样的语句,就可以得到服务端开放的interface接口对象,此时客户端可以直接调用这个对象的方法,犹如直接调用远程的Service对象一般。

四、Framework中使用AIDL

Framework中使用AIDL我们通过ITelephonyRegistry这个SystemService进行分析。该服务的主要作用就是对通话相关的事件进行监听,我们重心放在AIDL的实现结构上,不去关注ITelephonyRegistry具体的实现。

1、AIDL文件相关

先来看一下这个服务的AIDL文件:

  1. @ITelephonyRegistry.aidl
  2. interface ITelephonyRegistry {
  3. void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow);
  4. void notifyCallState(int state, String incomingNumber);
  5. void notifyServiceState(in ServiceState state);
  6. void notifySignalStrength(in SignalStrength signalStrength);
  7. void notifyMessageWaitingChanged(boolean mwi);
  8. }

再来看这个服务的真正实现:

  1. @TelephonyRegistry.java
  2. class TelephonyRegistry extends ITelephonyRegistry.Stub {
  3. TelephonyRegistry(Context context) {
  4. ......
  5. }
  6. void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow){
  7. ......
  8. }
  9. void notifyCallState(int state, String incomingNumber){
  10. ......
  11. }
  12. void notifyServiceState(in ServiceState state){
  13. ......
  14. }
  15. void notifySignalStrength(in SignalStrength signalStrength){
  16. ......
  17. }
  18. void notifyMessageWaitingChanged(boolean mwi){
  19. ......
  20. }
  21. }

上面的两个文件是这个服务的核心部分,aidl文件规定了这个服务的功能,而java文件是对功能的具体实现。但是,此时的TelephonyRegistry并没有继承Service的类,也就是说,当前他并不具备作为一个Service的资格。那么他是如何变成一个服务的呢?

2、服务的注册过程

在SystemService中可以找到答案。

  1. @SystemServer.java
  2. class ServerThread extends Thread {
  3. @Override
  4. public void run() {
  5. try {
  6. telephonyRegistry = new TelephonyRegistry(context);
  7. ServiceManager.addService("telephony.registry", telephonyRegistry);
  8. }
  9. }
  10. }

我们看到,在这一步中,把telephonyRegistry对象(也就是ITelephonyRegistry.Stub的子类对象)作为一个Service注册给了ServiceManager。并且注册的名字是“telephony.registry”
        有了这一步,TelephonyRegistry就可以作为服务提供者向客户端开放了。也就是说,有了这一步,TelephonyRegistry才算上是一个真正的Service,可以接受客户端的连接申请。

那么接下来,我们怎么得到这个Service呢?

3、如何得到注册的服务

既然通过ServiceManager注册了服务,我们就需要再次通过ServiceManager得到它的服务对象。

  1. private ITelephonyRegistry sRegistry;
  2. sRegistry = ITelephonyRegistry.Stub.asInterface(ServiceManager.getService("telephony.registry"));

通过这样的方法,我们就得到了ITelephonyRegistry.aidl的对象sRegistry。

4、流程总结

回顾一下这种AIDL服务的框架:通过继承ITelephonyRegistry.Stub父类去实现AIDL中规定的接口,然后把TelephonyRegistry作为ServiceManager注册给SystemService。客户端可以通过ServiceManager得到TelephonyRegistry的对象,然后就可以去调用AIDL中定义的接口了。

 

原文地址:https://www.cnblogs.com/linghu-java/p/8794389.html

时间: 2024-10-09 04:33:25

Android AIDL的用法的相关文章

android AIDL 语言用法

跨进程通信可以用AIDL语言 这里讲述下如何使用AIDL语言进行跨进程通信 文章参考 <设计模式>一书 demo结构参考 主要的文件类有:IBankAidl.aidl java文件:AidlBankBinder,BackActivity(应该是BankActivity写错了),BankService(继承自Service,服务类) IBankAidl.aidl文件 这里AIdl的使用对包位置有要求,所以我就把包名放出来了 package finishdemo.arcturis.binderan

Android AIDL使用详解

1.什么是aidl:aidl是 Android Interface definition language的缩写,一看就明白,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口 icp:interprocess communication :内部进程通信 2.既然aidl可以定义并实现进程通信,那么我们怎么使用它呢?文档/android-sdk/docs/guide/developing/tools/aidl.html中对步骤作了详细描述: --1.Create

Android AIDL 小结

1.AIDL (Android Interface Definition Language ) 2.AIDL 适用于 进程间通信,并且与Service端多个线程并发的情况,如果只是单个线程 可以使用 Messenger ,如果不需要IPC 可以使用Binder 3.AIDL语法:基础数据类型都可以适用,List Map等有限适用.static field 不适用. 4.AIDL基本用法 第一步:实现.aidl文件 接口描述文件 1.导入的包名 2.如果有使用Object对象,需要该对象 impl

Android aidl Binder框架浅析

转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38461079 ,本文出自[张鸿洋的博客] 1.概述 Binder能干什么?Binder可以提供系统中任何程序都可以访问的全局服务.这个功能当然是任何系统都应该提供的,下面我们简单看一下Android的Binder的框架 Android Binder框架分为服务器接口.Binder驱动.以及客户端接口:简单想一下,需要提供一个全局服务,那么全局服务那端即是服务器接口,任何程序即客

Android WIFI 简单用法

随着Wifi的普及,在开发App的时候对wifi的考虑越来越多了.例如程序的升级在wifi下可以省很多流量,在通信软件中的视频通话.可以实现高画质的传输等等,Android提供了WifiManager类来帮助开发者们管理Wifi.下面就简单来说一下WifiManager的简单用法把. 权限: 为了使用WfiManager 我们需要在Androidmanifest.xml 加入权限: //本例中使用了前两个.具体请按照需要添加权限. <uses-permission android:name=&quo

Android multipartentity的用法

最近写一个程序,android手机端上传多个图片到asp.net服务器端,使用httpclient,在网上查到了使用multipartentity.上传测试时总是出现(500)Internal Server Error,最后查出原来是文件大小超出了服务器限制,在项目的web.config中添加: <system.web> <HttpRuntime maxRequestLength="409600"  executionTimeout="60" /&

Android之Fragment用法

本文翻译自Android developer网站上面,原文参考:Building a Dynamic UI with Fragments 当我们需要动态的多界面切换的时候,就需要将UI元素和Activity融合成一个模块.在2.3中我们一般通过各种Activity中进行跳转来实现多界面的跳转和单个界面动态改变.在4.0或以上系统中就可以使用新的特性来方便的达到这个效果--Fragment类.Fragment类似一个嵌套Activity,可以定义自己的layout和自己的生命周期. 多个Fragm

Android TabHost(简易用法)

前言 欢迎大家我分享和推荐好用的代码段~~ 声明          欢迎转载,但请保留文章原始出处: CSDN:http://www.csdn.net 雨季o莫忧离:http://blog.csdn.net/luckkof 正文 Tab应用的结构 TabHost的Activity的结构如下: <?xml version="1.0" encoding="utf-8"?> <!-- 定义TabHost组件 --> <LinearLayout

Android之Adapter用法总结

http://blog.csdn.net/fznpcy/article/details/8658155 Android之Adapter用法总结 1.概念 Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带.在常见的View(List View,Grid View)等地方都需要用到Adapter.如下图直观的表达了Data.Adapter.View三者的关系: Android中所有的Adapter一览: 由图可以看到在Android中与Adapter有关