什么是aidl:
aidl是 Android Interface definition language的縮寫,一看就明白,它是一種android內(nèi)部進(jìn)程通信接口的描述語言,通過它我們可以定義進(jìn)程間的通信接口
icp:interprocess communication :內(nèi)部進(jìn)程通信。
在Android中, 每個(gè)應(yīng)用程序都有自己的進(jìn)程,當(dāng)需要在不同的進(jìn)程之間傳遞對(duì)象時(shí),該如何實(shí)現(xiàn)呢? 顯然, Java中是不支持跨進(jìn)程內(nèi)存共享的。因此要傳遞對(duì)象, 需要把對(duì)象解析成操作系統(tǒng)能夠理解的數(shù)據(jù)格式, 以達(dá)到跨界對(duì)象訪問的目的。在JavaEE中,采用RMI通過序列化傳遞對(duì)象。在Android中, 則采用AIDL(Android Interface Definition Language:接口定義語言)方式實(shí)現(xiàn)。
(1)在EcliPSe Android工程的Java包目錄中建立一個(gè)擴(kuò)展名為aidl的文件。該文件的語法類似于Java代碼,但會(huì)稍有不同。
(2)如果aidl文件的內(nèi)容是正確的,ADT會(huì)自動(dòng)生成一個(gè)Java接口文件(*.java)。
(3)建立一個(gè)服務(wù)類(Service的子類)。
(4)實(shí)現(xiàn)由aidl文件生成的Java接口。
(5)在AndroidManifest.xml文件中配置AIDL服務(wù),尤其要注意的是,<action>標(biāo)簽中android:name的屬性值就是客戶端要引用該服務(wù)的ID,也就是Intent類的參數(shù)值。
AIDL是一種接口定義語言,用于約束兩個(gè)進(jìn)程間的通訊規(guī)則,供編譯器生成代碼,實(shí)現(xiàn)Android設(shè)備上的兩個(gè)進(jìn)程間通信(IPC)。AIDL的IPC機(jī)制和EJB所采用的CORBA很類似,進(jìn)程之間的通信信息,首先會(huì)被轉(zhuǎn)換成AIDL協(xié)議消息,然后發(fā)送給對(duì)方,對(duì)方收到AIDL協(xié)議消息后再轉(zhuǎn)換成相應(yīng)的對(duì)象。由于進(jìn)程之間的通信信息需要雙向轉(zhuǎn)換,所以android采用代理類在背后實(shí)現(xiàn)了信息的雙向轉(zhuǎn)換,代理類由android編譯器生成,對(duì)開發(fā)人員來說是透明的。
實(shí)現(xiàn)進(jìn)程通信,一般需要下面四個(gè)步驟:
假設(shè)A應(yīng)用需要與B應(yīng)用進(jìn)行通信,調(diào)用B應(yīng)用中的download(String path)方法,B應(yīng)用以Service方式向A應(yīng)用提供服務(wù)。需要下面四個(gè)步驟:
1> 在B應(yīng)用中創(chuàng)建*.aidl文件,aidl文件的定義和接口的定義很相類,如:在cn.itcast.aidl包下創(chuàng)建IDownloadService.aidl文件,內(nèi)容如下:
package cn.itcast.aidl;
interface IDownloadService {
void download(String path);
}
當(dāng)完成aidl文件創(chuàng)建后,eclipse會(huì)自動(dòng)在項(xiàng)目的gen目錄中同步生成IDownloadService.java接口文件。接口文件中生成一個(gè)Stub的抽象類,里面包括aidl定義的方法,還包括一些其它輔助方法。值得關(guān)注的是asInterface(IBinder iBinder),它返回接口類型的實(shí)例,對(duì)于遠(yuǎn)程服務(wù)調(diào)用,遠(yuǎn)程服務(wù)返回給客戶端的對(duì)象為代理對(duì)象,客戶端在onServiceConnected(ComponentName name, IBinder service)方法引用該對(duì)象時(shí)不能直接強(qiáng)轉(zhuǎn)成接口類型的實(shí)例,而應(yīng)該使用asInterface(IBinder iBinder)進(jìn)行類型轉(zhuǎn)換。
編寫Aidl文件時(shí),需要注意下面幾點(diǎn):
1.接口名和aidl文件名相同。
2.接口和方法前不用加訪問權(quán)限修飾符public,private,protected等,也不能用final,static。
3.Aidl默認(rèn)支持的類型包話java基本類型(int、long、boolean等)和(String、List、Map、CharSequence),使用這些類型時(shí)不需要import聲明。對(duì)于List和Map中的元素類型必須是Aidl支持的類型。如果使用自定義類型作為參數(shù)或返回值,自定義類型必須實(shí)現(xiàn)Parcelable接口。
4.自定義類型和AIDL生成的其它接口類型在aidl描述文件中,應(yīng)該顯式import,即便在該類和定義的包在同一個(gè)包中。
5.在aidl文件中所有非Java基本類型參數(shù)必須加上in、out、inout標(biāo)記,以指明參數(shù)是輸入?yún)?shù)、輸出參數(shù)還是輸入輸出參數(shù)。
6.Java原始類型默認(rèn)的標(biāo)記為in,不能為其它標(biāo)記。
2> 在B應(yīng)用中實(shí)現(xiàn)aidl文件生成的接口(本例是IDownloadService),但并非直接實(shí)現(xiàn)接口,而是通過繼承接口的Stub來實(shí)現(xiàn)(Stub抽象類內(nèi)部實(shí)現(xiàn)了aidl接口),并且實(shí)現(xiàn)接口方法的代碼。內(nèi)容如下:
public class ServiceBinder extends IDownloadService.Stub {
@Override
public void download(String path) throws RemoteException {
Log.i("DownloadService", path);
}
}
3> 在B應(yīng)用中創(chuàng)建一個(gè)Service(服務(wù)),在服務(wù)的onBind(Intent intent)方法中返回實(shí)現(xiàn)了aidl接口的對(duì)象(本例是ServiceBinder)。內(nèi)容如下:
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);
}
}
}
其他應(yīng)用可以通過隱式意圖訪問服務(wù),意圖的動(dòng)作可以自定義,AndroidManifest.xml配置代碼如下:
<service android:name=".DownloadService" >
<intent-filter>
<action android:name="cn.itcast.process.aidl.DownloadService" />
</intent-filter>
</service>
4> 把B應(yīng)用中aidl文件所在package連同aidl文件一起拷貝到客戶端A應(yīng)用,eclipse會(huì)自動(dòng)在A應(yīng)用的gen目錄中為aidl文件同步生成IDownloadService.java接口文件,接下來就可以在A應(yīng)用中實(shí)現(xiàn)與B應(yīng)用通信,代碼如下:
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);//綁定到服務(wù)
}
@Override
protected void onDestroy() {
super.onDestroy();
this.unbindService(serviceConnection);//解除服務(wù)
}
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默認(rèn)支持的類型包話java基本類型(int、long、boolean等)和(String、List、Map、CharSequence),如果要傳遞自定義的類型該如何實(shí)現(xiàn)呢?
進(jìn)程間傳遞自定義類型的實(shí)現(xiàn)過程請(qǐng)參見頁面下方備注欄:
1> 創(chuàng)建自定義類型,并實(shí)現(xiàn)Parcelable接口,使其支持parcelable協(xié)議。如:在cn.itcast.domain包下創(chuàng)建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中的數(shù)據(jù)寫到Parcel
dest.writeInt(this.id);
dest.writestring(this.name);
}
//添加一個(gè)靜態(tài)成員,名為CREATOR,該對(duì)象實(shí)現(xiàn)了Parcelable.Creator接口
public static final Parcelable.Creator<Person> CREATOR = new Parcelable.Creator<Person>(){
@Override
public Person createFromParcel(Parcel source) {//從Parcel中讀取數(shù)據(jù),返回person對(duì)象
return new Person(source.readInt(), source.readString());
}
@Override
public Person[] newArray(int size) {
return new Person[size];
}
};
}
2> 在自定義類型所在包下創(chuàng)建一個(gè)aidl文件對(duì)自定義類型進(jìn)行聲明,文件的名稱與自定義類型同名。
package cn.itcast.domain;
parcelable Person;
3> 在接口aidl文件中使用自定義類型,需要使用import顯式導(dǎo)入,本例在cn.itcast.aidl包下創(chuàng)建IPersonService.aidl文件,內(nèi)容如下:
package cn.itcast.aidl;
import cn.itcast.domain.Person;
interface IPersonService {
void save(in Person person);
}
4> 在實(shí)現(xiàn)aidl文件生成的接口(本例是IPersonService),但并非直接實(shí)現(xiàn)接口,而是通過繼承接口的Stub來實(shí)現(xiàn)(Stub抽象類內(nèi)部實(shí)現(xiàn)了aidl接口),并且實(shí)現(xiàn)接口方法的代碼。內(nèi)容如下:
public class ServiceBinder extends IPersonService.Stub {
@Override
public void save(Person person) throws RemoteException {
Log.i("PersonService", person.getId()+"="+ person.getName());
}
}
5> 創(chuàng)建一個(gè)Service(服務(wù)),在服務(wù)的onBind(Intent intent)方法中返回實(shí)現(xiàn)了aidl接口的對(duì)象(本例是ServiceBinder)。內(nèi)容如下:
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());
}
}
}
其他應(yīng)用可以通過隱式意圖訪問服務(wù),意圖的動(dòng)作可以自定義,AndroidManifest.xml配置代碼如下:
<service android:name=".PersonService" >
<intent-filter>
<action android:name="cn.itcast.process.aidl.PersonService " />
</intent-filter>
</service>
6> 把應(yīng)用中的aidl文件和所在package一起拷貝到客戶端應(yīng)用的src目錄下,eclipse會(huì)自動(dòng)在客戶端應(yīng)用的gen目錄中為aidl文件同步生成IPersonService.java接口文件,接下來再把自定義類型文件和類型聲明aidl文件及所在package一起拷貝到客戶端應(yīng)用的src目錄下。
最后就可以在客戶端應(yīng)用中實(shí)現(xiàn)與遠(yuǎn)程服務(wù)的通信,代碼如下:
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);//綁定到服務(wù)
}
@Override
protected void onDestroy() {
super.onDestroy();
this.unbindService(serviceConnection);//解除服務(wù)
}
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;
}
};
}
最后我們?cè)诳偨Y(jié)一下android中 本地服務(wù)和 AIDL服務(wù)的區(qū)別:
本地服務(wù)不支持onBind(),它從onBind()返回null,這種類型的服務(wù)只能由承載服務(wù)的應(yīng)用程序組件訪問?梢哉{(diào)用 startService()來調(diào)用本地服務(wù)。AIDL服務(wù)可以同時(shí)供 同一進(jìn)程內(nèi)的組件和其他應(yīng)用程序的組件使用。這種類型的服務(wù)在AIDL 文件中為自身與其客戶端定義一個(gè)契約。服務(wù)實(shí)現(xiàn) AIDL契約,而客戶端綁定到 AIDL定義。服務(wù)通過從 onBind()方法 返回AIDL接口的實(shí)現(xiàn),來實(shí)現(xiàn)契約?蛻舳送ㄟ^調(diào)用 bindService()來綁定到AIDL服務(wù),并調(diào)用 unBindService()來從服務(wù)斷開。