一:AIDL,Android Interface definition language的缩写,它是一种android内部进程通信接口的描述语言,通过它我们可以定义进程间的通信接口。最近看了下AIDL在Android系统中的用法,在网上看到很多初学的朋友不太明白AIDL的实际作用,android提供了很多进程间通信的组件,像action、broadcast、contentprovide都可以实现进程间的通信,为什么还要用AIDL这个东西呢?
有开发过蓝牙或者WIFI应用的朋友肯定都知道,要去操作它必须先获得一个管理类,比如WIFI的管理类是WifiManager,通过getSystemService(Context.WIFI_SERVICE)就可以得到wifi的管理权限,这个提供了很多的方法可以让用户去操作它,比如打开wifi可以调用setWifiEnabled(true)方法。那这个Manager到底做了什么工作呢?是怎样实现打开wifi的呢?其实这个Manager只是一个管理类,真正干活的另有其人,是一个叫WifiService的系统服务。在Android系统中有很多的Manager,wifi的管理类叫WifiManager,蓝牙的管理类叫BluetoothManager,但是,只要有xxxManager.java,就会有Ixxx.aidl,并且有xxxService.java。这个aidl类就是实现Manager和Service通信的桥梁。
下面我们以源码中的WIFI为例来讲述一下AIDL在源码中的使用:
首先在这个目录下是已定义好的一个AIDL文件
frameworks/base/wifi/java/android/net/wifi/IWifiManager.aidl
关于AIDL的语言规范我就不多说了,其实和Java写接口差不多(它本来就是一种接口语言)。
然后在frameworks/base/Android.mk中添加一句:wifi/java/android/net/wifi/IWifiManager.aidl。android系统的编译目标是通过Android.mk来指定的,在这里加上自定义的aidl文件系统才会把这个文件编译进去,最终生成一个叫IWifiManager.java的文件。这一部分用eclipse可以很直观的看到,在项目添加了aidl文件后eclipse会自动编译此aidl,生成的文件存放在gen目录下。
AIDL添加好了后,定义了一个service继承IWifiManager这个aidl并实现了其中的方法。
frameworks/opt/net/wifi/service/java/com/android/server/wifi/WifiServiceImpl.java
然后在下面这个类中包装了一下
frameworks/opt/net/wifi/service/java/com/android/server/wifi/WifiService.java
然后在frameworks/base/core/java/android/content/Context.java文件中加入一个静态字符串:
public static final String WIFI_SERVICE = "wifi";
最后这个服务在SystemServer中启动
frameworks/base/services/java/com/android/server/SystemServer.java
private static final String WIFI_SERVICE_CLASS =
"com.android.server.wifi.WifiService";
mSystemServiceManager.startService(WIFI_SERVICE_CLASS);
SystemService是android系统跑起来之后就会调用的,这里的意思是把WifiService添加到系统服务中去,并取名字叫Context.WIFI_SERVICE,也就是wifi。这样看起来是不是很熟悉呢?每个开发人员在开发过程中肯定会调用系统服务的,比如电源管理服务:getSystemService(Context.POWER_SERVICE),这个Service也是在这里添加进去的。android有很多的系统服务,这里就不一一例举了,有兴趣的朋友可以自行看看这个文件。
现在aidl添加了,service也添加了,并且添加进了系统服务,那么还少一个Manager,可以让第三方程序调用的Manager。
frameworks/base/wifi/java/android/net/wifi/WifiManager.java
/**
* Create a new WifiManager instance.
* Applications will almost always want to use
* {@link android.content.Context#getSystemService Context.getSystemService()} to retrieve
* the standard {@link android.content.Context#WIFI_SERVICE Context.WIFI_SERVICE}.
* @param context the application context
* @param service the Binder interface
* @hide - hide this because it takes in a parameter of type IWifiManager, which
* is a system private class.
*/
public WifiManager(Context context, IWifiManager service, Looper looper) {
mContext = context;
mService = service;
mLooper = looper;
mTargetSdkVersion = context.getApplicationInfo().targetSdkVersion;
}
这个Manager的实例化是通过下面的方法实现的
(WifiManager) context.getSystemService(Context.WIFI_SERVICE)
这个系统的方法和通过aidl的Stub获取service是相通的,然后在这里去调用这个service的方法,以操作service,这就是aidl的作用。
二:源码编译AIDL Android.mk的写法:
LOCAL_SRC_FILES := $(call all-java-files-under)
LOCAL_SRC_FILES += \
src/com/asus/cnfindphone/service/persistentdata/IDataBlockService.aidl