前言
進(jìn)程間通信即IPC(Inter-Process Communication)。
安卓工程的manifest文件中可以指定各個(gè)組件運(yùn)行的進(jìn)程名(process)丽焊。
安卓的進(jìn)程間通信主要以service為基礎(chǔ),官方文檔
IPC有兩種方式浅蚪,使用Messenger和AIDL皮胡。
使用Messenger
Messenger是進(jìn)程間通信最簡單的實(shí)現(xiàn)方式,它在單一線程中處理所有消息請(qǐng)求拆檬,無須考慮線程安全問題洪己。 Messenger的基礎(chǔ)是Handler,使用步驟如下:
自定義一個(gè)Handler用于處理消息竟贯;
以自定義的Handler實(shí)例創(chuàng)建一個(gè)Messenger實(shí)例答捕, 在Service的IBinder onBind(Intent intent)方法返回該Messenger實(shí)例;
調(diào)用者bindService成功后屑那,在onServiceConnected(ComponentName className, IBinder service)中用Binder創(chuàng)建一個(gè)Messenger對(duì)象拱镐,Messenger messenger = new Messenger(service)艘款;
使用messenger對(duì)象發(fā)送消息對(duì)象Message,將消息交給自定義的Handler處理痢站,實(shí)現(xiàn)IPC
public class LocalService extends Service {
private final IBinder mBinder = new LocalBinder();
private final Random mGenerator = new Random();
public class LocalBinder extends Binder {
LocalService getService() {
return LocalService.this;
}
}
@Override
public IBinder onBind(Intent intent) {
return mBinder;
}
public int getRandomNumber() {
return mGenerator.nextInt(100);
}
}
/** 客戶端使用 */
public class BindingActivity extends Activity {
LocalService mService;
boolean mBound = false;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
}
@Override
protected void onStart() {
super.onStart();
Intent intent = new Intent(this, LocalService.class);
bindService(intent, mConnection, Context.BIND_AUTO_CREATE);
}
@Override
protected void onStop() {
super.onStop();
if (mBound) {
unbindService(mConnection);
mBound = false;
}
}
public void onButtonClick(View v) {
if (mBound) {
int num = mService.getRandomNumber();
Toast.makeText(this, "number: " + num, Toast.LENGTH_SHORT).show();
}
}
private ServiceConnection mConnection = new ServiceConnection() {
@Override
public void onServiceConnected(ComponentName className,
IBinder service) {
LocalBinder binder = (LocalBinder) service;
mService = binder.getService();
mBound = true;
}
@Override
public void onServiceDisconnected(ComponentName arg0) {
mBound = false;
}
};
}
使用AIDL
使用步驟如下磷箕,
- 新建一個(gè)aidl文件如IRemoteInterface.aidl,定義需要的接口阵难,build之后會(huì)生成一個(gè)同名的IRemoteInterface.class文件;
- IRemoteInterface中有一個(gè)Stub接口岳枷,接口與我們定義的一致,在Service中實(shí)現(xiàn)該Stub接口呜叫,并在onBind中返回該實(shí)現(xiàn)類實(shí)例空繁;
- 客戶端綁定service成功后,
public void onServiceConnected(ComponentName className, IBinder service) {
//可用這個(gè)訪問aidl中定義的接口
IRemoteInterface mIRemoteService = IRemoteService.Stub.asInterface(service);
}
上面Stub.asInterface方法中朱庆,會(huì)返回一個(gè)Stub類的代理對(duì)象盛泡,所以后續(xù)客戶端的方法調(diào)用實(shí)際是交給代理對(duì)象執(zhí)行的。
參數(shù)方向說明
aidl的參數(shù)有in, out , inout三個(gè)方向類型娱颊。因?yàn)榭邕M(jìn)程無法直接訪問對(duì)象結(jié)構(gòu)傲诵,對(duì)象的傳遞是通過序列化,所以自定義類作為參數(shù)要實(shí)現(xiàn)Parcelable接口箱硕,也要在aidl中聲明拴竹。正確聲明參數(shù)方向可以減少不必要的序列化操作,提高效率剧罩。
in即輸入?yún)?shù)栓拜,在當(dāng)前進(jìn)程序列化,遠(yuǎn)程進(jìn)程反序列化后得到對(duì)應(yīng)的參數(shù)值惠昔。
out為輸出參數(shù)幕与,遠(yuǎn)程進(jìn)程方法執(zhí)行完成后結(jié)果如果需要回傳,則進(jìn)行序列化寫入镇防,客戶端進(jìn)程進(jìn)行反序列化得到結(jié)果啦鸣。
inout為輸入輸出參數(shù),同時(shí)具備上面兩個(gè)的特性营罢。
Messenger和AIDL的區(qū)別
Messenger和AIDL都用于進(jìn)程間通信赏陵,Messenger使用較為簡單,內(nèi)部是在單一線程中處理消息請(qǐng)求的饲漾,線程安全蝙搔。AIDL則不是線程安全的。根據(jù)實(shí)際需要考传,
如果您想讓服務(wù)同時(shí)處理多個(gè)請(qǐng)求吃型,則可直接使用 AIDL。 在此情況下僚楞,您的服務(wù)必須具備多線程處理能力勤晚,并采用線程安全式設(shè)計(jì)枉层。否則使用Messenger
End.