Android 使用【AIDL】調(diào)用外部服務(wù)的解決方法
在Android 中有一種服務(wù)說(shuō)是服務(wù)其實(shí)倒不如說(shuō)是一個(gè)接口,這個(gè)接口名為:Android Interface Definition Language ,這個(gè)接口可提供跨進(jìn)程訪問(wèn)服務(wù),英文縮寫(xiě)為:AIDL。
此種服務(wù)的好處在于,多個(gè)應(yīng)用程序之間建立共同的服務(wù)機(jī)制,通過(guò)AIDL在不同應(yīng)用程序之間達(dá)到數(shù)據(jù)的共享和數(shù)據(jù)相互操作,下面將通過(guò)一個(gè)DEMO 演示AIDL 是如何為應(yīng)用程序之間提供服務(wù)的。
本文大綱為:
•1、創(chuàng)建AIDL 服務(wù)端。
•2、創(chuàng)建AIDL 客戶端。
•3、客戶端調(diào)用服務(wù)端提供的服務(wù)接口。
•4、小結(jié)。
本文要實(shí)現(xiàn)的功能大致如下:創(chuàng)建AIDL服務(wù)端,此服務(wù)端將提供一個(gè)Student 的javabean 提供客戶端取得數(shù)據(jù),因?yàn)閍idl 支持的數(shù)據(jù)類型比較簡(jiǎn)單,故這里建議把常用的數(shù)據(jù)類型的數(shù)據(jù)寫(xiě)入服務(wù)。
1、創(chuàng)建AIDL 服務(wù)端
在Android 的src 文件夾下的任意包里面新建文件,后綴名為*.aidl,如下圖
輸入如下代碼:
package com.aidl.test;
import com.aidl.test.Student;
interface IMyService
{
Map getMap(in String test_class,in Student student);
Student getStudent();
}
Student 類是一個(gè)序列化的類,這里使用Parcelable 接口來(lái)序列化是Google 提供的一個(gè)比Serializable 效率更高的序列化類。Student 類代碼如下:
package com.aidl.test;
import android.os.Parcel;
import android.os.Parcelable;
public class Student implements Parcelable {
private int age;
private String name;
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public static final Parcelable.Creator<Student> CREATOR = new Creator<Student>() {
@Override
public Student[] newArray(int size) {
// TODO Auto-generated method stub
return new Student[size];
}
@Override
public Student createFromParcel(Parcel source) {
// TODO Auto-generated method stub
return new Student(source);
}
};
public Student() {
}
public Student(Parcel pl) {
age = pl.readInt();
name = pl.readString();
}
@Override
public int describeContents() {
// TODO Auto-generated method stub
return 0;
}
@Override
public void writeToParcel(Parcel dest, int flags) {
// TODO Auto-generated method stub
dest.writeInt(age);
dest.writeString(name);
}
}
在這里必須注意,編寫(xiě)javabean時(shí)必須注意如下三點(diǎn):
•在Student 類中必須有一個(gè)靜態(tài)常量,常量名必須是CREATOR,而且CREATOR 常量的數(shù)據(jù)類型必須是 Parcelable.Creator
•在writeToParcel 方法中需要將要序列化的值寫(xiě)入到 Parcel對(duì)象中。
•編寫(xiě)完Student 為時(shí),必須再新建一個(gè)Student.aidl 文件,此文件輸入以下內(nèi)容:
parcelable Student; 這里的書(shū)寫(xiě)是供上面我們說(shuō)過(guò)的接口 *.aidl 文件導(dǎo)包時(shí)可以找到,并通過(guò)此文件找到Student類對(duì)象。
如果上面的步驟順利通過(guò)的話,Android 將會(huì)自動(dòng)在gen 目錄下R文件的相同目錄生成一個(gè)以*.aidl 文件命名的*.java 文件,如下圖:
順利生成成功后,我們?cè)賮?lái)編寫(xiě)一個(gè)AIDL 服務(wù)類,代碼如下:
package com.aidl.test;
import java.util.HashMap;
import java.util.Map;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.RemoteException;
public class MyService extends Service {
@Override
public IBinder onBind(Intent intent) {
// TODO Auto-generated method stub
return new MyServiceimpl();
}
public class MyServiceimpl extends IMyService.Stub {
@Override
public Student getStudent() throws RemoteException {
// TODO Auto-generated method stub
Student st = new Student();
st.setAge(18);
st.setName("terry");
return st;
}
@Override
public Map getMap(String testClass, Student student)
throws RemoteException {
// TODO Auto-generated method stub
Map<String, Object> map = new HashMap<String, Object>();
map.put("class", "五年級(jí)");
map.put("age", student.getAge());
map.put("name", student.getName());
return map;
}
}
}
如上代碼,MyService 服務(wù)類有一個(gè)子類并繼承自我們上面生成的*.java 文件重寫(xiě)其中我們?cè)?.aidl 中聲明的兩個(gè)接口方法,實(shí)現(xiàn)了其功能。上面IBinder 必須返回此服務(wù)類的子類對(duì)象,否則客戶端將無(wú)法獲得服務(wù)對(duì)象。
最后,即然有服務(wù)的操作,那么就得在manifest文件中注冊(cè)服務(wù)類,代碼如下:
<service android:name=".MyService">
<intent-filter>
<action android:name="com.aidl.test.IMyService"></action>
</intent-filter>
</service>
至此,服務(wù)端就己經(jīng)開(kāi)發(fā)完成了,下面接著開(kāi)發(fā)客啟端。
2、創(chuàng)建AIDL 客戶端
同樣是新建一個(gè)項(xiàng)目,這里要注意,需要將服務(wù)端生成成功后的gen 目錄下的包復(fù)制過(guò)來(lái),放到我們新建項(xiàng)目的src 文件夾下,如下圖:
因?yàn)镮MyService 這個(gè)生成類,引用到了Student 這個(gè)javabean 所以這里一并將javabean也復(fù)制過(guò)來(lái)。
至此,客戶端的創(chuàng)建己經(jīng)完畢,下面我們就要利用創(chuàng)建的客戶端去調(diào)用服務(wù)端的方法。
3、客戶端調(diào)用服務(wù)端提供的服務(wù)接口
先看一下運(yùn)行效果:
細(xì)心的朋友會(huì)發(fā)現(xiàn),上面的數(shù)據(jù)不是我們?cè)谏厦婵蛻舳藶镾tudent 設(shè)置的數(shù)據(jù)嗎?怎么在這個(gè)程序 里面也同樣得到了?沒(méi)錯(cuò)。這就是aidl 的魅力,下面來(lái)看看如何調(diào)用 吧,圖中有兩個(gè)按鈕,一個(gè)按鈕為綁定AIDL 服務(wù),即通過(guò)Activity 的 bindService 綁定 AIDL 外部服務(wù),全部代碼如下:
package com.aidl.client;
import com.aidl.test.IMyService;
import android.app.Activity;
import android.app.AlertDialog;
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.os.RemoteException;
import android.view.View;
import android.view.View.OnClickListener;
import android.widget.Button;
public class aidlActivity extends Activity implements OnClickListener {
Button btn1, btn2;
private IMyService myService = null;
private ServiceConnection serviceConnection = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
// TODO Auto-generated method stub
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
// TODO Auto-generated method stub
myService = IMyService.Stub.asInterface(service);
btn2.setEnabled(true);
}
};
/** Called when the activity is first created. */
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
btn1 = (Button) findViewById(R.id.Button01);
btn2 = (Button) findViewById(R.id.Button02);
btn2.setEnabled(false);
btn1.setOnClickListener(this);
btn2.setOnClickListener(this);
}
@Override
public void onClick(View v) {
// TODO Auto-generated method stub
switch (v.getId()) {
case R.id.Button01:
bindService(new Intent("com.aidl.test.IMyService"),
serviceConnection, Context.BIND_AUTO_CREATE);
break;
case R.id.Button02:
StringBuilder sb = new StringBuilder();
try {
sb.append("學(xué)生名稱為:" + myService.getStudent().getName() + "\n");
sb.append("年齡為:" + myService.getStudent().getAge() + "\n");
sb.append("map 對(duì)象內(nèi)容為如下:"
+ myService.getMap("中國(guó)", myService.getStudent())
.toString());
} catch (RemoteException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
new AlertDialog.Builder(aidlActivity.this).setTitle("調(diào)用外部服務(wù)")
.setMessage(sb.toString()).setPositiveButton(
android.R.string.ok, null).show();
break;
default:
break;
}
}
}
在ServiceConnetction里面對(duì)IMyService 進(jìn)行初始化,即可操作該對(duì)象 ,該對(duì)象就可以得到我們所有要處理的數(shù)據(jù)。
4、小結(jié)
•aidl 文件調(diào)用javabean 的aidl文件必須導(dǎo)包;
•javabean 必須序列化,如果沒(méi)有用javabean可以用簡(jiǎn)單的變量代替,如返回一個(gè)整型,返回一個(gè)字符串等。
•使用aidl 必須同時(shí)存在客戶端和服務(wù)端,即客戶端在本機(jī)上,服務(wù)端也在本機(jī)上,要使用客戶端必須服務(wù)端事先在本機(jī)上注冊(cè)過(guò)服務(wù)。
代碼下載:服務(wù)端DEMO
客戶端DEMO
相關(guān)文章
Webview實(shí)現(xiàn)android簡(jiǎn)單的瀏覽器實(shí)例代碼
這篇文章主要介紹了Webview實(shí)現(xiàn)android簡(jiǎn)單的瀏覽器實(shí)例代碼的相關(guān)資料,需要的朋友可以參考下2016-02-02Android基于廣播事件機(jī)制實(shí)現(xiàn)簡(jiǎn)單定時(shí)提醒功能代碼
這篇文章主要介紹了Android基于廣播事件機(jī)制實(shí)現(xiàn)簡(jiǎn)單定時(shí)提醒功能代碼,較為詳細(xì)的分析了Android廣播事件機(jī)制及提醒功能的相關(guān)實(shí)現(xiàn)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10Flutter 實(shí)現(xiàn)酷炫的3D效果示例代碼
這篇文章主要介紹了Flutter 實(shí)現(xiàn)酷炫的3D效果,本文通過(guò)實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-07-07Android studio點(diǎn)擊跳轉(zhuǎn)WebView詳解
這篇文章主要為大家詳細(xì)介紹了Android studio點(diǎn)擊跳轉(zhuǎn)WebView的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09Android利用GridView實(shí)現(xiàn)單選功能
這篇文章主要為大家詳細(xì)介紹了Android利用GridView實(shí)現(xiàn)單選功能的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-02-02Android 日常開(kāi)發(fā)總結(jié)的60條技術(shù)經(jīng)驗(yàn)
這篇文章主要介紹了Android日常開(kāi)發(fā)總結(jié)的技術(shù)經(jīng)驗(yàn)60條,需要的朋友可以參考下2016-03-03Android 中 EventBus 的使用之多線程事件處理
這篇文章主要介紹了Android 中 EventBus 的使用之多線程事件處理的相關(guān)資料,需要的朋友可以參考下2015-11-11