引言
應(yīng)用程序組件有一個(gè)生命周期——一開(kāi)始android實(shí)例化他們響應(yīng)意圖,直到結(jié)束實(shí)例被銷毀。在這期間,他們有時(shí)候處于激活狀態(tài),有時(shí)候處于非激活狀 態(tài);對(duì)于活動(dòng),對(duì)用戶有時(shí)候可見(jiàn),有時(shí)候不可見(jiàn)。組件生命周期將討論活動(dòng)、服務(wù)、廣播接收者的生命周期——包括在生命周期中他們可能的狀態(tài)、通知狀態(tài)改變 的方法、及這些狀態(tài)的組件寄宿的進(jìn)程被終結(jié)和實(shí)例被銷毀的可能性。
上篇android開(kāi)發(fā)之旅:組件生命周期(一)講解了論活動(dòng)的生命周期及他們可能的狀態(tài)、通知狀態(tài)改變的方法。本篇將介紹服務(wù)和廣播接收者的生命周期:
服務(wù)生命周期
廣播接收者生命周期
1、服務(wù)生命周期
一個(gè)服務(wù)可以用在兩個(gè)方面:
它可以啟動(dòng)且允許一直運(yùn)行直到有人停止它,或者它自己停止。在這種模式,通過(guò)調(diào)用context.startservice()啟動(dòng)服務(wù)及通過(guò)調(diào)用 context.stopservice()停止服務(wù)。服務(wù)也可以通過(guò)調(diào)用service.stopself()或 service.stopselfresult()停止自己。僅需要調(diào)用一次stopservice()停止服務(wù),而不管調(diào)用 startservice()了多少次。
通過(guò)使用相關(guān)接口可以編程地操作服務(wù)。客戶端建立與service對(duì)象的一個(gè)連接及使用該連接調(diào)入服務(wù)。連接通過(guò)調(diào)用 context.bindservice()建立,通過(guò)調(diào)用context.unbindservice()關(guān)閉。多個(gè)客戶端可以綁定到同一個(gè)服務(wù)。如果 服務(wù)尚未啟動(dòng),bindservice()可以選擇啟動(dòng)它。
這兩種模式并不是完全分離的。你可以綁定到一個(gè)用startservice()啟動(dòng)的服務(wù)。例如,一個(gè)后臺(tái)音樂(lè)服務(wù)可以通過(guò)使用定義了音樂(lè)播放的 intent對(duì)象調(diào)用startservice()啟動(dòng)。直到后來(lái),用戶可能想對(duì)播放器做一些控制或者獲取當(dāng)前歌曲的一些信息,一個(gè)活動(dòng)將調(diào)用 bindservice()與服務(wù)建立連接。在這種情況下,實(shí)際上直到最后一個(gè)綁定關(guān)閉stopservice()并不會(huì)停止。
像活動(dòng)一樣,一個(gè)服務(wù)也有生命周期方法,你可以執(zhí)行監(jiān)視它的狀態(tài)改變。但是比活動(dòng)的生命周期方法更少,只有三個(gè)且它們是公有的(public)而不是受保護(hù)的(protected)(說(shuō)明:活動(dòng)的生命周期方法是protected的):
void oncreate()
void onstart(intent intent)
void ondestory()
通過(guò)這三個(gè)方法,你可以監(jiān)視服務(wù)生命周期的兩個(gè)嵌套循環(huán):
服務(wù)的整個(gè)生命時(shí)間(entire lifetime),從調(diào)用oncreate()到相應(yīng)地調(diào)用ondestory()。像一個(gè)活動(dòng)一樣,服務(wù)在oncreate()中做一些初始設(shè)置,且 在中釋放所有的資源。例如,一個(gè)音樂(lè)播放服務(wù)可以在oncreate()中創(chuàng)建線程,然后在ondestory()中停止線程。
服務(wù)的活躍生命時(shí)間(active lifetime),從調(diào)用onstart()開(kāi)始。這個(gè)方法傳遞參數(shù)是傳送給startservice()的intent對(duì)象。音樂(lè)服務(wù)將打開(kāi)intent,了解播放哪個(gè)音樂(lè)并且開(kāi)始播放。
沒(méi)有相應(yīng)的回調(diào)方法,因?yàn)榉?wù)停止沒(méi)有onstop()方法。
startservice()和ondestory()被所有服務(wù)調(diào)用,不管是通過(guò)context.startservice()啟動(dòng)還是通過(guò) context.bindservice()啟動(dòng)的。然而,onstart()僅被通過(guò)startservice()啟動(dòng)的服務(wù)調(diào)用。
如果一個(gè)服務(wù)允許別的綁定到它,有一些額外的回調(diào)方法來(lái)實(shí)現(xiàn)它:
ibinder onbind(intent intent)
boolean onunbind(intent intent)
void onrebind(intent intent)
onbind()回調(diào)傳遞的參數(shù)是傳給bindservice()的intent對(duì)象,onunbind()回調(diào)傳遞的參數(shù)是傳給 unbindservice()的intent對(duì)象。如果服務(wù)允許綁定,onbind()返回客戶端與服務(wù)交互的通信通道。onunbind()方法可以 要求調(diào)用onrebind(),如果一個(gè)新的客戶端連接到服務(wù)。
下圖解釋了服務(wù)的回調(diào)方法。雖然,它分離了由startservice()啟動(dòng)的服務(wù)和由bindservice()啟動(dòng)的服務(wù),記住任何服務(wù),無(wú)論 它怎么啟動(dòng)的,都可能允許客戶端綁定到它,因此任何服務(wù)可能接收onbind()和onunbind()調(diào)用。
可以發(fā)現(xiàn)第一次startservice時(shí),會(huì)調(diào)用oncreate和onstart,在沒(méi)有stopservice前,無(wú)論點(diǎn)擊多少次 startservice,都只會(huì)調(diào)用onstart。而stopservice時(shí)調(diào)用ondestroy。再次點(diǎn)擊stopservice,會(huì)發(fā)現(xiàn)不會(huì) 進(jìn)入service的生命周期的,即不會(huì)再調(diào)用oncreate,onstart和ondestroy。
而onbind在startservice/stopservice中沒(méi)有調(diào)用。
需要注意一個(gè)問(wèn)題,當(dāng)activity退出的時(shí)候,sercvice并不會(huì)停止,此時(shí)我們可以再進(jìn)入activity重新綁定,當(dāng)這時(shí)候service 就會(huì)調(diào)用onrebind()方法,但是調(diào)用onrebind()方法的前提是先前的onunbind()方法執(zhí)行成功,但是使用 super.onunbind(intent)是執(zhí)行不成功的,這時(shí)候我們要手動(dòng)的使其返回true,再次綁定時(shí)rebind()就會(huì)執(zhí)行。否則,如果退 出時(shí)不顯示的指定onunbind()為成功的話(為false),那么重新啟動(dòng)此activity來(lái)綁定服務(wù)時(shí),service的onbind()方法 和onrebind都不會(huì)執(zhí)行,但是serviceconnection方法確一定會(huì)回調(diào)了。這說(shuō)明在service中的onbind()方法不同于 onstart()方法不能被重復(fù)調(diào)用。
2、廣播接收者生命周期
一個(gè)廣播接收者有一個(gè)回調(diào)方法:void onreceive(context curcontext, intent broadcastmsg)。當(dāng)一個(gè)廣播消息到達(dá)接收者是,android調(diào)用它的onreceive()方法并傳遞給它包含消息的intent對(duì)象。廣 播接收者被認(rèn)為僅當(dāng)它執(zhí)行這個(gè)方法時(shí)是活躍的。當(dāng)onreceive()返回后,它是不活躍的。
有一個(gè)活躍的廣播接收者的進(jìn)程是受保護(hù)的,不會(huì)被殺死。但是系統(tǒng)可以在任何時(shí)候殺死僅有不活躍組件的進(jìn)程,當(dāng)占用的內(nèi)存別的進(jìn)程需要時(shí)。
這帶來(lái)一個(gè)問(wèn)題,當(dāng)一個(gè)廣播消息的響應(yīng)時(shí)費(fèi)時(shí)的,因此應(yīng)該在獨(dú)立的線程中做這些事,遠(yuǎn)離用戶界面其它組件運(yùn)行的主線程。如果onreceive()衍 生線程然后返回,整個(gè)進(jìn)程,包括新的線程,被判定為不活躍的(除非進(jìn)程中的其它應(yīng)用程序組件是活躍的),將使它處于被殺的危機(jī)。解決這個(gè)問(wèn)題的方法是 onreceive()啟動(dòng)一個(gè)服務(wù),及時(shí)服務(wù)做這個(gè)工作,因此系統(tǒng)知道進(jìn)程中有活躍的工作在做。
ps:service 類兩種啟動(dòng)方法:
• context.startservice()
• context.bindservice()
1. 在同一個(gè)應(yīng)用任何地方調(diào)用 startservice() 方法就能啟動(dòng) service 了,然后系統(tǒng)會(huì)回調(diào) service 類的 oncreate() 以及 onstart() 方法。這樣啟動(dòng)的 service 會(huì)一直運(yùn)行在后臺(tái),直到 context.stopservice() 或者 selfstop() 方法被調(diào)用。
另外如果一個(gè) service 已經(jīng)被啟動(dòng),其他代碼再試圖調(diào)用 startservice() 方法,是不會(huì)執(zhí)行 oncreate() 的,但會(huì)重新執(zhí)行一次 onstart() 。
2. 另外一種 bindservice() 方法的意思是,把這個(gè) service 和調(diào)用 service 的客戶類綁起來(lái),如果調(diào)用這個(gè)客戶類被銷毀,service 也會(huì)被銷毀。用這個(gè)方法的一個(gè)好處是,bindservice() 方法執(zhí)行后 service 會(huì)回調(diào)上邊提到的 onbind() 方發(fā),你可以從這里返回一個(gè)實(shí)現(xiàn)了 ibind 接口的類,在客戶端操作這個(gè)類就能和這個(gè)服務(wù)通信了,比如得到 service 運(yùn)行的狀態(tài)或其他操作。如果 service 還沒(méi)有運(yùn)行,使用這個(gè)方法啟動(dòng) service 就會(huì) oncreate() 方法而不會(huì)調(diào)用 onstart()。
總結(jié):
1. startservice()的目的是回調(diào)onstart()方法,oncreate() 方法是在service不存在的時(shí)候調(diào)用的,如果service存在(例如之前調(diào)用了bindservice,那么service的oncreate方法已經(jīng)調(diào)用了)那么startservice()將跳過(guò)oncreate() 方法。
2. bindservice()目的是回調(diào)onbind()方法,它的作用是在service和調(diào)用者之間建立一個(gè)橋梁,并不負(fù)責(zé)更多的工作(例如一個(gè)service需要連接服務(wù)器的操作),一般使用bindservice來(lái)綁定到一個(gè)現(xiàn)有的service(即通過(guò)startservice啟動(dòng)的服務(wù))。
由于service 的onstart()方法只有在startservice()啟動(dòng)service的情況下才調(diào)用,故使用onstart()的時(shí)候要注意這點(diǎn)。