Android AIDL的用法
一、什么是AIDL服务
二、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:
文件内容:
- package du.pack;
- interface IMyService{
- //只有一个接口
- String getValue();
- }
3.2.2、创建服务类及实现内部类
- public class MyService extends Service {
- @Override
- public IBinder onBind(Intent arg0) {
- // 把内部类的对象返回给客户端使用
- return new MyServiceImpl();
- }
- // 创建一个继承自IMyService.Stub的内部类
- public class MyServiceImpl extends IMyService.Stub {
- // 必须实现AIDL文件中的接口
- public String getValue() throws RemoteException {
- return null;
- }
- }
- }
注意,我们写的service中,onBind方法必须返回MyServiceImpl类的对象实例,否则客户端无法获得服务对象。
3.2.3、在AndroidManifest.xml文件中配置MyService类
- <service android:name=".MyService" >
- <intent-filter>
- <action android:name="du.pack.IMyService" />
- </intent-filter>
- </service>
上面的"du.pack.IMyService"是客户端用于访问AIDL服务的ID。
4、本地客户端的使用方法
4.1、新建一个Eclipse Android工程,并将刚才远程服务端自动生成的gen目录下面的IMyService.java文件连同包目录一起复制到客户端工程的src目录中。
4.2、调用AIDL服务首先要绑定服务,然后才能获得服务对象。
- public class AidlClientTestActivity extends Activity {
- // 远程服务端的对象
- IMyService mIMyService;
- private ServiceConnection mConnection = new ServiceConnection() {
- public void onServiceConnected(ComponentName name, IBinder service) {
- // 绑定成功,得到远程服务端的对象,目标完成!!!
- mIMyService = IMyService.Stub.asInterface(service);
- }
- public void onServiceDisconnected(ComponentName name) {
- // 解除绑定
- mIMyService = null;
- }
- };
- @Override
- public void onCreate(Bundle savedInstanceState) {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- // 绑定远程服务端服务
- Intent serviceIntent = new Intent("du.pack.IMyService");
- bindService(serviceIntent, mConnection, Context.BIND_AUTO_CREATE);
- }
- }
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文件:
- @ITelephonyRegistry.aidl
- interface ITelephonyRegistry {
- void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow);
- void notifyCallState(int state, String incomingNumber);
- void notifyServiceState(in ServiceState state);
- void notifySignalStrength(in SignalStrength signalStrength);
- void notifyMessageWaitingChanged(boolean mwi);
- }
再来看这个服务的真正实现:
- @TelephonyRegistry.java
- class TelephonyRegistry extends ITelephonyRegistry.Stub {
- TelephonyRegistry(Context context) {
- ......
- }
- void listen(String pkg, IPhoneStateListener callback, int events, boolean notifyNow){
- ......
- }
- void notifyCallState(int state, String incomingNumber){
- ......
- }
- void notifyServiceState(in ServiceState state){
- ......
- }
- void notifySignalStrength(in SignalStrength signalStrength){
- ......
- }
- void notifyMessageWaitingChanged(boolean mwi){
- ......
- }
- }
上面的两个文件是这个服务的核心部分,aidl文件规定了这个服务的功能,而java文件是对功能的具体实现。但是,此时的TelephonyRegistry并没有继承Service的类,也就是说,当前他并不具备作为一个Service的资格。那么他是如何变成一个服务的呢?
2、服务的注册过程
在SystemService中可以找到答案。
- @SystemServer.java
- class ServerThread extends Thread {
- @Override
- public void run() {
- try {
- telephonyRegistry = new TelephonyRegistry(context);
- ServiceManager.addService("telephony.registry", telephonyRegistry);
- }
- }
- }
我们看到,在这一步中,把telephonyRegistry对象(也就是ITelephonyRegistry.Stub的子类对象)作为一个Service注册给了ServiceManager。并且注册的名字是“telephony.registry”
有了这一步,TelephonyRegistry就可以作为服务提供者向客户端开放了。也就是说,有了这一步,TelephonyRegistry才算上是一个真正的Service,可以接受客户端的连接申请。
那么接下来,我们怎么得到这个Service呢?
3、如何得到注册的服务
既然通过ServiceManager注册了服务,我们就需要再次通过ServiceManager得到它的服务对象。
- private ITelephonyRegistry sRegistry;
- sRegistry = ITelephonyRegistry.Stub.asInterface(ServiceManager.getService("telephony.registry"));
通过这样的方法,我们就得到了ITelephonyRegistry.aidl的对象sRegistry。
4、流程总结
回顾一下这种AIDL服务的框架:通过继承ITelephonyRegistry.Stub父类去实现AIDL中规定的接口,然后把TelephonyRegistry作为ServiceManager注册给SystemService。客户端可以通过ServiceManager得到TelephonyRegistry的对象,然后就可以去调用AIDL中定义的接口了。
Android AIDL的用法的更多相关文章
- android AIDL 语言用法
跨进程通信可以用AIDL语言 这里讲述下如何使用AIDL语言进行跨进程通信 文章参考 <设计模式>一书 demo结构参考 主要的文件类有:IBankAidl.aidl java文件:Aid ...
- Android aidl Binder框架浅析
转载请标明出处:http://blog.csdn.net/lmj623565791/article/details/38461079 ,本文出自[张鸿洋的博客] 1.概述 Binder能干什么?B ...
- Android AIDL 小结
1.AIDL (Android Interface Definition Language ) 2.AIDL 适用于 进程间通信,并且与Service端多个线程并发的情况,如果只是单个线程 可以使用 ...
- 【Android学习】android:layout_weight的用法实例
对于android:layout_weight的用法,用下面的例子来说明: <LinearLayout xmlns:android="http://schemas.android.co ...
- Android之Adapter用法总结-(转)
Android之Adapter用法总结 1.概念 Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带.在常见的View(List View,Grid Vie ...
- Android之Adapter用法总结(转)
Android之Adapter用法总结 1.概念 Adapter是连接后端数据和前端显示的适配器接口,是数据和UI(View)之间一个重要的纽带.在常见的View(List View,Grid Vie ...
- Android webservice的用法详细讲解
Android webservice的用法详细讲解 看到有很多朋友对WebService还不是很了解,在此就详细的讲讲WebService,争取说得明白吧.此文章采用的项目是我毕业设计的webserv ...
- Android AIDL自动生成Java文件测试
/******************************************************************************** * Android AIDL自动生成 ...
- Using self-defined Parcelable objects during an Android AIDL RPC / IPC call
Using self-defined Parcelable objects during an Android AIDL RPC / IPC call In my previous post “Usi ...
随机推荐
- Mysql Cluster节点类型(转载)
mysql 自4.1.x版本开始推出mysql cluster功能.cluster简单地说,就是一组"节点"的组合.这里的节点是一个逻辑概念,一台计算机上可以存放一个节点,也可以存 ...
- python开发_python中的变量:全局变量和局部变量
如果你在为python中的变量:全局变量和局部变量头疼,我想这篇blog会给你帮助 运行效果: 代码部分: #Python中的变量:全局变量和局部变量 #在很多语言中,在声明全局变量的时候,都喜欢把全 ...
- 02_java之数据类型和基本运算
01变量概述 * A: 什么是变量? * a: 变量是一个内存中的小盒子(小容器),容器是什么?生活中也有很多容器,例如水杯是容器,用来装载水:你家里的大衣柜是容器,用来装载衣裤:饭盒是容器,用来装载 ...
- 六、配置github的pull request触发jenkins自动构建
之前的配置,都是向master分支push操作触发jenkins进行构建,但是在一般的正常工作中,不会允许程序员直接向主分支推送代码:正常都是fork一个本地的分支,在本地分支调试完后,向主干分支提交 ...
- SQLite的基本用法
SQLite是Android自带的轻量级数据库,接口封装的很好,不会SQL的也能很好的使用. 接下来讲一下怎么创建数据库.通过adb查看数据表和数据.增删查改. 一.创建数据库 Android封装了S ...
- 使用JAVA爬取京东商品价格
有一件物品,你想看看它在京东下所有搜索结果的价格,要怎么办呢? 京东这个网站还是很好爬的,所有价格信息都写在了Html里面,而且跳到第二页之后,url也是有规律的,基本没有什么技术难度. 例如:想找i ...
- IIS Manager 配置文件修该,允许跨域CORS访问
IIS Manager 配置文件修该,允许跨域CORS访问 IIS Manager 的api访问会出现跨域问题,需要 IIS Manager的配置文件中修改. 配置文件的路径:C:\Program F ...
- 从零玩转JavaWeb系列7web服务器-----get与post的区别
总结get与post的区别 get参数通过url传递,post放在request body中. get请求在url中传递的参数是有长度限制的,而post没有. get比post更不安全,因为参数直接暴 ...
- lucene3.0范围查找
在lucene3.0以上版本中,范围查询也有很大的变化,RangeQuery已经不推荐使用,使用TermRangeQuery和NumericRangeQuery两个替代.TermRangeQuery: ...
- 前端 webpack
前端 webpack http://www.cnblogs.com/lvdabao/