建立AIDL服務(wù)的步驟(1)
建立AIDL服務(wù)要比建立普通的服務(wù)復(fù)雜一些,具體步驟如下:
(1)在Eclipse Android工程的Java包目錄中建立一個(gè)擴(kuò)展名為aidl的文件。該文件的語法類似于Java代碼,但會(huì)稍有不同。詳細(xì)介紹見實(shí)例52的內(nèi)容。
(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ù)值。這一點(diǎn)將在實(shí)例52和實(shí)例53中看到。
實(shí)例52:建立AIDL服務(wù)
AIDL服務(wù)工程目錄:src\ch08\ch08_aidl
客戶端程序工程目錄:src\ch08\ch08_aidlclient
本例中將建立一個(gè)簡(jiǎn)單的AIDL服務(wù)。這個(gè)AIDL服務(wù)只有一個(gè)getValue方法,該方法返回一個(gè)String類型的值。在安裝完服務(wù)后,會(huì)在客戶端調(diào)用這個(gè)getValue方法,并將返回值在TextView組件中輸出。建立這個(gè)AIDL服務(wù)的步驟如下:
(1)建立一個(gè)aidl文件。在Java包目錄中建立一個(gè)IMyService.aidl文件。IMyService.aidl文件的位置如圖8.24所示。
|
圖8.24 IMyService.aidl文件的位置 |
IMyService.aidl文件的內(nèi)容如下:
- package net.blogjava.mobile.aidl;
- interface IMyService
- {
- String getValue();
- }
IMyService.aidl文件的內(nèi)容與Java代碼非常相似,但要注意,不能加修飾符(例如,public、private)、AIDL服務(wù)不支持的數(shù)據(jù)類型(例如,InputStream、OutputStream)等內(nèi)容。
(2)如果IMyService.aidl文件中的內(nèi)容輸入正確,ADT會(huì)自動(dòng)生成一個(gè)IMyService.java文件。讀者一般并不需要關(guān)心這個(gè)文件的具體內(nèi)容,也不需要維護(hù)這個(gè)文件。關(guān)于該文件的具體內(nèi)容,讀者可以查看本節(jié)提供的源代碼。
(3)編寫一個(gè)MyService類。MyService是Service的子類,在MyService類中定義了一個(gè)內(nèi)嵌類(MyServiceImpl),該類是IMyService.Stub的子類。MyService類的代碼如下:
- package net.blogjava.mobile.aidl;
-
- import android.app.Service;
- import android.content.Intent;
- import android.os.IBinder;
- import android.os.RemoteException;
-
- public class MyService extends Service
- {
- public class MyServiceImpl extends IMyService.Stub
- {
- @Override
- public String getValue() throws RemoteException
- {
- return "Android/OPhone開發(fā)講義";
- }
- }
- @Override
- public IBinder onBind(Intent intent)
- {
- return new MyServiceImpl();
- }
- }
在編寫上面代碼時(shí)要注意如下兩點(diǎn):
IMyService.Stub是根據(jù)IMyService.aidl文件自動(dòng)生成的,一般并不需要管這個(gè)類的內(nèi)容,只需要編寫一個(gè)繼承于IMyService.Stub類的子類(MyServiceImpl類)即可。
onBind方法必須返回MyServiceImpl類的對(duì)象實(shí)例,否則客戶端無法獲得服務(wù)對(duì)象。
(4)在AndroidManifest.xml文件中配置MyService類,代碼如下:
- <service android:name=".MyService" >
- <intent-filter>
- <action android:name="net.blogjava.mobile.aidl.IMyService" />
- </intent-filter>
- </service>
其中"net.blogjava.mobile.aidl.IMyService"是客戶端用于訪問AIDL服務(wù)的ID。
下面來編寫客戶端的調(diào)用代碼。首先新建一個(gè)Eclipse Android工程(ch08_aidlclient),并將自動(dòng)生成的IMyService.java文件連同包目錄一起復(fù)制到ch08_aidlclient工程的src目錄中,如圖8.25所示。
|
圖8.25 IMyService.java文件
在ch08_aidlclient工程中的位置 |
調(diào)用AIDL服務(wù)首先要綁定服務(wù),然后才能獲得服務(wù)對(duì)象,代碼如下:
- package net.blogjava.mobile;
-
- import net.blogjava.mobile.aidl.IMyService;
- import android.app.Activity;
- import android.content.ComponentName;
- import android.content.Context;
- import android.content.Intent;
- import android.content.ServiceConnection;
- import android.os.Bundle;
- import android.os.IBinder;
- import android.view.View;
- import android.view.View.OnClickListener;
- import android.widget.Button;
- import android.widget.TextView;
-
- public class Main extends Activity implements OnClickListener
- {
- private IMyService myService = null;
- private Button btnInvokeAIDLService;
- private Button btnBindAIDLService;
- private TextView textView;
- private ServiceConnection serviceConnection =
new ServiceConnection()
- {
- @Override
- public void onServiceConnected(ComponentName
name, IBinder service)
- {
- // 獲得服務(wù)對(duì)象
- myService = IMyService.Stub.asInterface(service);
- btnInvokeAIDLService.setEnabled(true);
- }
- @Override
- public void onServiceDisconnected(ComponentName name)
- {
- }
- };
- @Override
- public void onClick(View view)
- {
- switch (view.getId())
- {
- case R.id.btnBindAIDLService:
- // 綁定AIDL服務(wù)
- bindService(new Intent("net.blogjava.
mobile.aidl.IMyService"),
- serviceConnection, Context.BIND_AUTO_CREATE);
- break;
- case R.id.btnInvokeAIDLService:
- try
- {
- textView.setText(myService.
getValue()); // 調(diào)用服務(wù)端的getValue方法
- }
- catch (Exception e)
- {
- }
- break;
- }
- }
- @Override
- public void onCreate(Bundle savedInstanceState)
- {
- super.onCreate(savedInstanceState);
- setContentView(R.layout.main);
- btnInvokeAIDLService = (Button) findViewById
(R.id.btnInvokeAIDLService);
- btnBindAIDLService = (Button) findViewById
(R.id.btnBindAIDLService);
- btnInvokeAIDLService.setEnabled(false);
- textView = (TextView) findViewById(R.id.textview);
- btnInvokeAIDLService.setOnClickListener(this);
- btnBindAIDLService.setOnClickListener(this);
- }
- }