溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Service詳解

發(fā)布時(shí)間:2020-06-10 09:43:57 來源:網(wǎng)絡(luò) 閱讀:165 作者:yym631 欄目:移動(dòng)開發(fā)

service可以在和多場(chǎng)合的應(yīng)用中使用,比如播放多媒體的時(shí)候用戶啟動(dòng)了其他Activity這個(gè)時(shí)候程序要在后臺(tái)繼續(xù)播放,比如檢測(cè)SD卡上文件的變化,再或者在后臺(tái)記錄你地理信息位置的改變等等,總之服務(wù)嘛,總是藏在后頭的。


Service是在一段不定的時(shí)間運(yùn)行在后臺(tái),不和用戶交互應(yīng)用組件。每個(gè)Service必須在manifest中 通過<service>來聲明。可以通過contect.startservice和contect.bindserverice來啟動(dòng)。


Service和其他的應(yīng)用組件一樣,運(yùn)行在進(jìn)程的主線程中。這就是說如果service需要很多耗時(shí)或者阻塞的操作,需要在其子線程中實(shí)現(xiàn)。


service的兩種模式(startService()/bindService()不是完全分離的):


    Context.startService(Intent intent) 

    Context.bindService(Intent intent, ServiceConnection conn, int flags)

    

    stopService(Intent intent)

    unbindService(ServiceConnection conn)

    

    1. 使用startService()方法啟用服務(wù),調(diào)用者與服務(wù)之間沒有關(guān)連,即使調(diào)用者退出了,服務(wù)仍然運(yùn)行。

    2. 使用bindService()方法啟用服務(wù),調(diào)用者與服務(wù)綁定在了一起,調(diào)用者一旦退出,服務(wù)也就終止,大有“不求同時(shí)生,必須同時(shí)死”的特點(diǎn)。

----------------------------------------------------------    

本地服務(wù) Local Service 用于應(yīng)用程序內(nèi)部。


    它可以啟動(dòng)并運(yùn)行,直至有人停止了它或它自己停止。在這種方式下,它以調(diào)用Context.startService()啟動(dòng),而以調(diào)用Context.stopService()結(jié)束。它可以調(diào)用Service.stopSelf() 或 Service.stopSelfResult()來自己停止。不論調(diào)用了多少次startService()方法,你只需要調(diào)用一次stopService()來停止服務(wù)。


  用于實(shí)現(xiàn)應(yīng)用程序自己的一些耗時(shí)任務(wù),比如查詢升級(jí)信息,并不占用應(yīng)用程序比如Activity所屬線程,而是單開線程后臺(tái)執(zhí)行,這樣用戶體驗(yàn)比較好。


遠(yuǎn)程服務(wù) Remote Service 用于android系統(tǒng)內(nèi)部的應(yīng)用程序之間。


    它可以通過自己定義并暴露出來的接口進(jìn)行程序操作。客戶端建立一個(gè)到服務(wù)對(duì)象的連接,并通過那個(gè)連接來調(diào)用服務(wù)。連接以調(diào)用Context.bindService()方法建立,以調(diào)用 Context.unbindService()關(guān)閉。多個(gè)客戶端可以綁定至同一個(gè)服務(wù)。如果服務(wù)此時(shí)還沒有加載,bindService()會(huì)先加載它。


  可被其他應(yīng)用程序復(fù)用,比如天氣預(yù)報(bào)服務(wù),其他應(yīng)用程序不需要再寫這樣的服務(wù),調(diào)用已有的即可。    

-----------------------------------------------------------------    


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的客戶類綁起來,如果調(diào)用這個(gè)客戶類被銷毀,Service也會(huì)被銷毀。用這個(gè)方法的一個(gè)好處是,bindService()方法執(zhí)行后Service會(huì)回調(diào)onBind()方發(fā),它的作用是在Service和調(diào)用者之間建立一個(gè)橋梁,并不負(fù)責(zé)更多的工作,你可以從這里返回一個(gè)實(shí)現(xiàn)了IBind接口的類,在客戶端操作這個(gè)類就能和這個(gè)服務(wù)通信了,比如得到Service運(yùn)行的狀態(tài)或其他操作。如果Service還沒有運(yùn)行,使用這個(gè)方法啟動(dòng)Service就會(huì)onCreate()方法而不會(huì)調(diào)用onStart()。一般使用bindService來綁定到一個(gè)現(xiàn)有的Service(即通過StartService啟動(dòng)的服務(wù))。


-----------------------------------------------------------------------


與Service通信并且讓它持續(xù)運(yùn)行

    如果我們想保持和Service的通信,又不想讓Service隨著Activity退出而退出呢?你可以先startService()然后再bindService()。當(dāng)你不需要綁定的時(shí)候就執(zhí)行unbindService()方法,執(zhí)行這個(gè)方法只會(huì)觸發(fā)Service的onUnbind()而不會(huì)把這個(gè)Service銷毀。這樣就可以既保持和Service的通信,也不會(huì)隨著Activity銷毀而銷毀了。



提高 Service 優(yōu)先級(jí)

    Android 系統(tǒng)對(duì)于內(nèi)存管理有自己的一套方法,為了保障系統(tǒng)有序穩(wěn)定的運(yùn)信,系統(tǒng)內(nèi)部會(huì)自動(dòng)分配,控制程序的內(nèi)存使用。當(dāng)系統(tǒng)覺得當(dāng)前的資源非常有限的時(shí)候,為了保證一些優(yōu)先級(jí)高的程序能運(yùn)行,就會(huì)殺掉一些他認(rèn)為不重要的程序或者服務(wù)來釋放內(nèi)存。這樣就能保證真正對(duì)用戶有用的程序仍然再運(yùn)行。如果你的 Service 碰上了這種情況,多半會(huì)先被殺掉。但如果你增加 Service 的優(yōu)先級(jí)就能讓他多留一會(huì),我們可以用setForeground(true) 來設(shè)置Service 的優(yōu)先級(jí)。


    為什么是 foreground ? 默認(rèn)啟動(dòng)的Service是被標(biāo)記為background,當(dāng)前運(yùn)行的Activity一般被標(biāo)記為foreground,也就是說你給Service設(shè)置了foreground那么他就和正在運(yùn)行的Activity類似優(yōu)先級(jí)得到了一定的提高。當(dāng)讓這并不能保證你得Service永遠(yuǎn)不被殺掉,只是提高了他的優(yōu)先級(jí)。

    

    

    優(yōu)先級(jí)規(guī)則

    1. 如果service正在調(diào)用onCreate,onStartCommand或者onDestory方法,那么用于當(dāng)前service的進(jìn)程則變?yōu)榍芭_(tái)進(jìn)程以避免被killed。

    2. 如果當(dāng)前service已經(jīng)被啟動(dòng)(start),擁有它的進(jìn)程則比那些用戶可見的進(jìn)程優(yōu)先級(jí)低一些,但是比那些不可見的進(jìn)程更重要,這就意味著service一般不會(huì)被killed.

    3. 如果客戶端已經(jīng)連接到service(bindService),那么擁有Service的進(jìn)程則擁有最高的優(yōu)先級(jí),可以認(rèn)為service是可見的。

    4. 如果service可以使用startForeground(int, Notification)方法來將service設(shè)置為前臺(tái)狀態(tài),那么系統(tǒng)就認(rèn)為是對(duì)用戶可見的,并不會(huì)在內(nèi)存不足時(shí)killed。

    如果有其他的應(yīng)用組件作為Service,Activity等運(yùn)行在相同的進(jìn)程中,那么將會(huì)增加該進(jìn)程的重要性。

     

-----------------------------------------------------------------------------------


Service生命周


Android Service生命周期與Activity生命周期是相似的,但是也存在一些細(xì)節(jié)上也存在著重要的不同:


onCreate和onStart是不同的

    通過從客戶端調(diào)用Context.startService(Intent)方法我們可以啟動(dòng)一個(gè)服務(wù)。如果這個(gè)服務(wù)還沒有運(yùn)行,Android將啟動(dòng)它并且在onCreate方法之后調(diào)用它的onStart方法。如果這個(gè)服務(wù)已經(jīng)在運(yùn)行,那么它的onStart方法將被新的Intent再次調(diào)用。所以對(duì)于單個(gè)運(yùn)行的Service它的onStart方法被反復(fù)調(diào)用是完全可能的并且是很正常的。


onResume、onPause以及onStop是不需要的

    回調(diào)一個(gè)服務(wù)通常是沒有用戶界面的,所以我們也就不需要onPause、onResume或者onStop方法了。無論何時(shí)一個(gè)運(yùn)行中的Service它總是在后臺(tái)運(yùn)行。


onBind

    如果一個(gè)客戶端需要持久的連接到一個(gè)服務(wù),那么他可以調(diào)用Context.bindService方法。如果這個(gè)服務(wù)沒有運(yùn)行,將通過調(diào)用onCreate方法去創(chuàng)建這個(gè)服務(wù)但并不調(diào)用onStart方法來啟動(dòng)它。相反,onBind方法將被客戶端的Intent調(diào)用,并且它返回一個(gè)IBind對(duì)象以便客戶端稍后可以調(diào)用這個(gè)服務(wù)。同一服務(wù)被客戶端同時(shí)啟動(dòng)和綁定是很正常的。


onDestroy

    與Activity一樣,當(dāng)一個(gè)服務(wù)被結(jié)束是onDestroy方法將會(huì)被調(diào)用。當(dāng)沒有客戶端啟動(dòng)或綁定到一個(gè)服務(wù)時(shí)Android將終結(jié)這個(gè)服務(wù)。與很多Activity時(shí)的情況一樣,當(dāng)內(nèi)存很低的時(shí)候Android也可能會(huì)終結(jié)一個(gè)服務(wù)。如果這種情況發(fā)生,Android也可能在內(nèi)存夠用的時(shí)候嘗試啟動(dòng)被終止的服務(wù),所以你的服務(wù)必須為重啟持久保存信息,并且最好在onStart方法內(nèi)來做。


使用context.startService() 啟動(dòng)Service是會(huì)會(huì)經(jīng)歷:

  context.startService() ->onCreate()- >onStart()->Service running

  context.stopService() | ->onDestroy() ->Service stop

    如果Service還沒有運(yùn)行,則android先調(diào)用onCreate()然后調(diào)用onStart();如果Service已經(jīng)運(yùn)行,則只調(diào)用onStart(),所以一個(gè)Service的onStart方法可能會(huì)重復(fù)調(diào)用多次。


    stopService的時(shí)候直接onDestroy,如果是調(diào)用者自己直接退出而沒有調(diào)用stopService的話,Service會(huì)一直在后臺(tái)運(yùn)行。該Service的調(diào)用者再啟動(dòng)起來后可以通過stopService關(guān)閉Service。


  所以調(diào)用startService的生命周期為:onCreate --> onStart(可多次調(diào)用) --> onDestroy


使用使用context.bindService()啟動(dòng)Service會(huì)經(jīng)歷:

  context.bindService()->onCreate()->onBind()->Service running

  onUnbind() -> onDestroy() ->Service stop

    

    onBind將返回給客戶端一個(gè)IBind接口實(shí)例,IBind允許客戶端回調(diào)服務(wù)的方法,比如得到Service運(yùn)行的狀態(tài)或其他操作。這個(gè)時(shí)候把調(diào)用者(Context,例如Activity)會(huì)和Service綁定在一起,Context退出了,Srevice就會(huì)調(diào)用onUnbind->onDestroy相應(yīng)退出。


  所以調(diào)用bindService的生命周期為:onCreate --> onBind(只一次,不可多次綁定) --> onUnbind --> onDestory。


    在Service每一次的開啟關(guān)閉過程中,只有onStart可被多次調(diào)用(通過多次startService調(diào)用),其他onCreate,onBind,onUnbind,onDestory在一個(gè)生命周期中只能被調(diào)用一次。

----------------------------------------

Service和Thread的區(qū)別

我們拿服務(wù)來進(jìn)行一個(gè)后臺(tái)長(zhǎng)時(shí)間的動(dòng)作,為了不阻塞線程,然而,Thread就可以達(dá)到這個(gè)效果,為什么我們不直接使用Thread去代替服務(wù)呢?(這個(gè)問題摘抄至網(wǎng)上,原文地址不是是哪個(gè),所以沒寫上)

這里提下,

 

1). Thread:Thread 是程序執(zhí)行的最小單元,它是分配CPU的基本單位。可以用 Thread 來執(zhí)行一些異步的操作。

2). Service:Service 是android的一種機(jī)制,當(dāng)它運(yùn)行的時(shí)候如果是Local Service,那么對(duì)應(yīng)的 Service 是運(yùn)行在主進(jìn)程的 main 線程上的。如:onCreate,onStart 這些函數(shù)在被系統(tǒng)調(diào)用的時(shí)候都是在主進(jìn)程的 main 線程上運(yùn)行的。如果是Remote Service,那么對(duì)應(yīng)的 Service 則是運(yùn)行在獨(dú)立進(jìn)程的 main 線程上。因此請(qǐng)不要把 Service 理解成線程,它跟線程半毛錢的關(guān)系都沒有!

 

既然這樣,那么我們?yōu)槭裁匆?Service 呢?其實(shí)這跟 android 的系統(tǒng)機(jī)制有關(guān),我們先拿 Thread 來說。Thread 的運(yùn)行是獨(dú)立于 Activity 的,也就是說當(dāng)一個(gè) Activity 被 finish 之后,如果你沒有主動(dòng)停止 Thread 或者 Thread 里的 run 方法沒有執(zhí)行完畢的話,Thread 也會(huì)一直執(zhí)行。因此這里會(huì)出現(xiàn)一個(gè)問題:當(dāng) Activity 被 finish 之后,你不再持有該 Thread 的引用。另一方面,你沒有辦法在不同的 Activity 中對(duì)同一 Thread 進(jìn)行控制。


舉個(gè)例子:如果你的 Thread 需要不停地隔一段時(shí)間就要連接服務(wù)器做某種同步的話,該 Thread 需要在 Activity 沒有start的時(shí)候也在運(yùn)行。這個(gè)時(shí)候當(dāng)你 start 一個(gè) Activity 就沒有辦法在該 Activity 里面控制之前創(chuàng)建的 Thread。因此你便需要?jiǎng)?chuàng)建并啟動(dòng)一個(gè) Service ,在 Service 里面創(chuàng)建、運(yùn)行并控制該 Thread,這樣便解決了該問題(因?yàn)槿魏?Activity 都可以控制同一 Service,而系統(tǒng)也只會(huì)創(chuàng)建一個(gè)對(duì)應(yīng) Service 的實(shí)例)。


因此你可以把 Service 想象成一種消息服務(wù),而你可以在任何有 Context 的地方調(diào)用 Context.startService、Context.stopService、Context.bindService,Context.unbindService,來控制它,你也可以在 Service 里注冊(cè) BroadcastReceiver,在其他地方通過發(fā)送 broadcast 來控制它,當(dāng)然這些都是 Thread 做不到的。





<manifest ... >

  ...

  <application ... >

      <service android:name=".ExampleService" />

      ...

  </application>

</manifest>


向AI問一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI