Android執(zhí)行Service有兩種方法充蓝,一種是startService隧枫,一種是bindService。下面讓我們一起來聊一聊這兩種執(zhí)行Service方法的區(qū)別谓苟。
1官脓、生命周期上的區(qū)別
執(zhí)行startService時(shí),Service會(huì)經(jīng)歷onCreate->onStartCommand涝焙。當(dāng)執(zhí)行stopService時(shí)卑笨,直接調(diào)用onDestroy方法。調(diào)用者如果沒有stopService仑撞,Service會(huì)一直在后臺(tái)運(yùn)行赤兴,下次調(diào)用者再起來仍然可以stopService。
執(zhí)行bindService時(shí)隧哮,Service會(huì)經(jīng)歷onCreate->onBind桶良。這個(gè)時(shí)候調(diào)用者和Service綁定在一起。調(diào)用者調(diào)用unbindService方法或者調(diào)用者Context不存在了(如Activity被finish了)沮翔,Service就會(huì)調(diào)用onUnbind->onDestroy陨帆。這里所謂的綁定在一起就是說兩者共存亡了。
多次調(diào)用startService鉴竭,該Service只能被創(chuàng)建一次歧譬,即該Service的onCreate方法只會(huì)被調(diào)用一次岸浑。但是每次調(diào)用startService搏存,onStartCommand方法都會(huì)被調(diào)用。Service的onStart方法在API 5時(shí)被廢棄矢洲,替代它的是onStartCommand方法璧眠。
第一次執(zhí)行bindService時(shí),onCreate和onBind方法會(huì)被調(diào)用,但是多次執(zhí)行bindService時(shí)责静,onCreate和onBind方法并不會(huì)被多次調(diào)用袁滥,即并不會(huì)多次創(chuàng)建服務(wù)和綁定服務(wù)。
2灾螃、調(diào)用者如何獲取綁定后的Service的方法
onBind回調(diào)方法將返回給客戶端一個(gè)IBinder接口實(shí)例题翻,IBinder允許客戶端回調(diào)服務(wù)的方法,比如得到Service運(yùn)行的狀態(tài)或其他操作腰鬼。我們需要IBinder對(duì)象返回具體的Service對(duì)象才能操作角撞,所以說具體的Service對(duì)象必須首先實(shí)現(xiàn)Binder對(duì)象况芒。
3、既使用startService又使用bindService的情況
如果一個(gè)Service又被啟動(dòng)又被綁定,則該Service會(huì)一直在后臺(tái)運(yùn)行奕锌。首先不管如何調(diào)用,onCreate始終只會(huì)調(diào)用一次呛谜。對(duì)應(yīng)startService調(diào)用多少次溶诞,Service的onStart方法便會(huì)調(diào)用多少次。Service的終止拧篮,需要unbindService和stopService同時(shí)調(diào)用才行词渤。不管startService與bindService的調(diào)用順序,如果先調(diào)用unbindService串绩,此時(shí)服務(wù)不會(huì)自動(dòng)終止掖肋,再調(diào)用stopService之后,服務(wù)才會(huì)終止赏参;如果先調(diào)用stopService志笼,此時(shí)服務(wù)也不會(huì)終止,而再調(diào)用unbindService或者之前調(diào)用bindService的Context不存在了(如Activity被finish的時(shí)候)之后把篓,服務(wù)才會(huì)自動(dòng)停止纫溃。
那么,什么情況下既使用startService韧掩,又使用bindService呢紊浩?
如果你只是想要啟動(dòng)一個(gè)后臺(tái)服務(wù)長(zhǎng)期進(jìn)行某項(xiàng)任務(wù),那么使用startService便可以了疗锐。如果你還想要與正在運(yùn)行的Service取得聯(lián)系坊谁,那么有兩種方法:一種是使用broadcast,另一種是使用bindService滑臊。前者的缺點(diǎn)是如果交流較為頻繁口芍,容易造成性能上的問題,而后者則沒有這些問題雇卷。因此鬓椭,這種情況就需要startService和bindService一起使用了颠猴。
另外,如果你的服務(wù)只是公開一個(gè)遠(yuǎn)程接口小染,供連接上的客戶端(Android的Service是C/S架構(gòu))遠(yuǎn)程調(diào)用執(zhí)行方法翘瓮,這個(gè)時(shí)候你可以不讓服務(wù)一開始就運(yùn)行,而只是bindService裤翩,這樣在第一次bindService的時(shí)候才會(huì)創(chuàng)建服務(wù)的實(shí)例運(yùn)行它资盅,這會(huì)節(jié)約很多系統(tǒng)資源,特別是如果你的服務(wù)是遠(yuǎn)程服務(wù)踊赠,那么效果會(huì)越明顯(當(dāng)然在Servcie創(chuàng)建的是偶會(huì)花去一定時(shí)間律姨,這點(diǎn)需要注意)。
4臼疫、本地服務(wù)與遠(yuǎn)程服務(wù)
本地服務(wù)依附在主進(jìn)程上择份,在一定程度上節(jié)約了資源。本地服務(wù)因?yàn)槭窃谕贿M(jìn)程烫堤,因此不需要IPC荣赶,也不需要AIDL。相應(yīng)bindService會(huì)方便很多鸽斟。缺點(diǎn)是主進(jìn)程被kill后拔创,服務(wù)變會(huì)終止。
遠(yuǎn)程服務(wù)是獨(dú)立的進(jìn)程富蓄,對(duì)應(yīng)進(jìn)程名格式為所在包名加上你指定的android:process字符串剩燥。由于是獨(dú)立的進(jìn)程,因此在Activity所在進(jìn)程被kill的是偶立倍,該服務(wù)依然在運(yùn)行灭红。缺點(diǎn)是該服務(wù)是獨(dú)立的進(jìn)程,會(huì)占用一定資源口注,并且使用AIDL進(jìn)行IPC稍微麻煩一點(diǎn)变擒。
對(duì)于startService來說,不管是本地服務(wù)還是遠(yuǎn)程服務(wù)寝志,我們需要做的工作都一樣簡(jiǎn)單娇斑。
5、代碼實(shí)例
startService啟動(dòng)服務(wù)
public class LocalService1 extends Service {
/**
* onBind 是 Service 的虛方法材部,因此我們不得不實(shí)現(xiàn)它毫缆。
* 返回 null,表示客服端不能建立到此服務(wù)的連接乐导。
*/
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onCreate() {
super.onCreate();
}
@Override
public void onStartCommand(Intent intent, int startId, int flags) {
super.onStartCommand(intent, startId, flags);
}
@Override
public void onDestroy() {
super.onDestroy();
}
}
bindService綁定服務(wù)
public class LocalService extends Service {
/**
* 在 Local Service 中我們直接繼承 Binder 而不是 IBinder,因?yàn)?Binder 實(shí)現(xiàn)了 IBinder 接口苦丁,這樣我們可以** 少做很多工作。
*/
public class SimpleBinder extends Binder{
/**
* 獲取 Service 實(shí)例
* @return
*/
public LocalService getService(){
return LocalService.this;
}
public int add(int a, int b){
return a + b;
}
}
public SimpleBinder sBinder;
@Override
public void onCreate() {
super.onCreate();
// 創(chuàng)建 SimpleBinder
sBinder = new SimpleBinder();
}
@Override
public IBinder onBind(Intent intent) {
// 返回 SimpleBinder 對(duì)象
return sBinder;
}
}
上面的代碼關(guān)鍵之處兽叮,在于 onBind(Intent) 這個(gè)方法 返回了一個(gè)實(shí)現(xiàn)了 IBinder 接口的對(duì)象芬骄,這個(gè)對(duì)象將用于綁定Service 的 Activity 與 Local Service 通信。
下面是 Activity 中的代碼:
public class Main extends Activity {
private final static String TAG = "SERVICE_TEST";
private ServiceConnection sc;
private boolean isBind;
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.main);
sc = new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {
}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
LocalService.SimpleBinder sBinder = (LocalService.SimpleBinder)service;
Log.v(TAG, "3 + 5 = " + sBinder.add(3, 5));
Log.v(TAG, sBinder.getService().toString());
}
};
findViewById(R.id.btnBind).setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
bindService(new Intent(Main.this, LocalService.class), sc, Context.BIND_AUTO_CREATE);
isBind = true;
}
});
findViewById(R.id.btnUnbind).setOnClickListener(new OnClickListener() {
@Override
public void onClick(View v) {
if(isBind){
unbindService(sc);
isBind = false;
}
}
});
}
}
6鹦聪、在AndroidManifest.xml里Service元素常見選項(xiàng)
android:name ------------- 服務(wù)類名
android:label -------------- 服務(wù)的名字账阻,如果此項(xiàng)不設(shè)置,那么默認(rèn)顯示的服務(wù)名則為類名
android:icon -------------- 服務(wù)的圖標(biāo)
android:permission ------- 申明此服務(wù)的權(quán)限泽本,這意味著只有提供了該權(quán)限的應(yīng)用才能控制或連接此服務(wù)
android:process ---------- 表示該服務(wù)是否運(yùn)行在另外一個(gè)進(jìn)程淘太,如果設(shè)置了此項(xiàng),那么將會(huì)在包名后面加上這段字符串表示另一進(jìn)程的名字
android:enabled ---------- 表示是否能被系統(tǒng)實(shí)例化规丽,為true表示可以蒲牧,為false表示不可以,默認(rèn)為true
android:exported --------- 表示該服務(wù)是否能夠被其他應(yīng)用程序所控制或連接赌莺,不設(shè)置默認(rèn)此項(xiàng)為 false