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