Android?O對后臺(tái)Service限制詳解
Service問題
Service沒有界面,運(yùn)行于后臺(tái),它會(huì)消耗設(shè)備資源,并且可能會(huì)導(dǎo)致不好的用戶體驗(yàn),例如資源占用過多,導(dǎo)致設(shè)備運(yùn)行不流暢。為了緩解這個(gè)問題,Android O版本(Android 8.0, API 26)對后臺(tái)Service強(qiáng)加了一些限制。注意,只是對后臺(tái)Service加了限制,前臺(tái)Service不受影響。
什么是前臺(tái)應(yīng)用
在解釋后臺(tái)Service的限制之前,首先需要知道什么是前臺(tái)應(yīng)用,什么是后臺(tái)應(yīng)用。
一個(gè)前臺(tái)應(yīng)用必須滿足如下某一個(gè)條件:
- 有可見的Activity。無論是resume狀態(tài)還是pause狀態(tài),只要可見就行。
- 有前臺(tái)Service。
- 有其它app連接到當(dāng)前app,通過綁定Service或者使用ContentProvider。
前臺(tái)Service和后臺(tái)Service
那么前臺(tái)Service和后臺(tái)Service如何區(qū)分呢?
前臺(tái)Service會(huì)發(fā)送一條通知,讓用戶察覺到有一個(gè)Service正在運(yùn)行,而后臺(tái)Service沒有通知,用戶不會(huì)察覺到有一個(gè)Service正在運(yùn)行。
那么前臺(tái)Service和后臺(tái)Service是如何啟動(dòng)的呢?
對于后臺(tái)Service,很簡單,通過Context.startService()
啟動(dòng)的就是后臺(tái)Service。
對于前臺(tái)Service呢,這個(gè)有點(diǎn)復(fù)雜。在Android 8.0之前,首先通過Context.startService()
啟動(dòng)一個(gè)后臺(tái)Service,然后通過Service.startForeground()
發(fā)送一條通知,如此一來,后臺(tái)Service就變成了前臺(tái)Service。但是從Android 8.0開始,系統(tǒng)限制了后臺(tái)app創(chuàng)建后臺(tái)服務(wù),所以就無法再使用之前的方法把后臺(tái)Service提升到前臺(tái)。
為了解決后臺(tái)app無法創(chuàng)建前臺(tái)Service問題,Android 8.0又引入了一個(gè)新方法Context.startForegroundService()
來直接啟動(dòng)一個(gè)前臺(tái)Service,但是當(dāng)系統(tǒng)創(chuàng)建這個(gè)前臺(tái)Service后,應(yīng)用需要在5秒內(nèi)調(diào)用Service.startForeground()
來顯示一個(gè)通知,否則系統(tǒng)會(huì)停止這個(gè)前臺(tái)Service,并彈出ANR。
從Android 9.0開始(Android P, API 28),如果要?jiǎng)?chuàng)建前臺(tái)Service,還要在AndroidManifest.xml中聲明android.permission.FOREGROUND_SERVICE
權(quán)限,這是一個(gè)普通的權(quán)限,系統(tǒng)會(huì)自動(dòng)授予app。如果不這樣做,會(huì)拋出異常。
下面給出一個(gè)例子,如何創(chuàng)建前臺(tái)Service。
首先App的目標(biāo)版本是Android 10(API 29),那么需要在清單文件中聲明權(quán)限。
<uses-permission android:name="android.permission.FOREGROUND_SERVICE" />
然后在Activity中啟動(dòng)前臺(tái)Service,為了兼容Android 8.0之前的版本,代碼應(yīng)該如下
protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); mIntent = new Intent(this, MyService.class); if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.O) { startForegroundService(mIntent); } else { startService(mIntent); } }
我們還要記得在任務(wù)完成時(shí)銷毀Service,可以在Activity的onDestroy()
中調(diào)用stopService()
,也可以在Service中調(diào)用stopSelf()
。到底調(diào)用哪種需要根據(jù)具體要求來選擇。
在Service創(chuàng)建時(shí),我們需要調(diào)用Service.startForeground()
public void onCreate() { super.onCreate(); // 發(fā)送通知,把service置于前臺(tái) NotificationManagerCompat notificationManager = NotificationManagerCompat.from(this); // 從Android 8.0開始,需要注冊通知通道 if (android.os.Build.VERSION.SDK_INT >= android.os.Build.VERSION_CODES.O) { NotificationChannel notificationChannel = new NotificationChannel(NOTIFICATION_CHANNEL_ID, "Service notification channel", NotificationManager.IMPORTANCE_DEFAULT); notificationManager.createNotificationChannel(notificationChannel); } NotificationCompat.Builder builder = new NotificationCompat.Builder(this, NOTIFICATION_CHANNEL_ID) .setSmallIcon(R.mipmap.ic_launcher) .setContentTitle("Service"); // 注意第一個(gè)參數(shù)不能為0 startForeground(666, builder.build()); }
后臺(tái)Service限制
當(dāng)一個(gè)app處于前臺(tái)時(shí),它可以隨意創(chuàng)建和使用后臺(tái)服務(wù)。當(dāng)這個(gè)app進(jìn)入后臺(tái),它只有幾分鐘的窗口期可以創(chuàng)建和使用服務(wù)。當(dāng)這個(gè)窗口期結(jié)束后,系統(tǒng)認(rèn)為這個(gè)app進(jìn)入了空閑狀態(tài),此時(shí)系統(tǒng)會(huì)停止app的后臺(tái)服務(wù)。
解決后臺(tái)Service限制
官方建議使用JobScheduler替換后臺(tái)Service,官方還舉了一個(gè)例子,有一個(gè)圖庫app需要檢測當(dāng)前用戶是否收到了來自朋友分享的圖片,即使app沒有運(yùn)行于前臺(tái)。在Android 8.0之前,可以使用后臺(tái)Service來檢測應(yīng)用的云存儲(chǔ),但是這有一個(gè)問題,這個(gè)Service一起在后臺(tái)運(yùn)行,它會(huì)消耗資源,影響手機(jī)性能。但是從Android 8.0,使用JobScheduler替換后臺(tái)Service,它會(huì)周期性啟動(dòng)一個(gè)任務(wù),查詢服務(wù)器,然后退出。相比于后臺(tái)Service,它消耗的資源明顯較少,間接提升了手機(jī)性能。
如何使用JobScheduler呢,我將在下一篇文章分享一個(gè)下載圖片的案例。
參考文獻(xiàn)
官方文檔 developer.android.google.cn/about/versi…
以上就是Android O對后臺(tái)Service限制詳解的詳細(xì)內(nèi)容,更多關(guān)于Android O對后臺(tái)Service限制的資料請關(guān)注腳本之家其它相關(guān)文章!
- Android?Service啟動(dòng)綁定流程詳解
- Android布局控件View?ViewRootImpl?WindowManagerService關(guān)系
- android?微信搶紅包工具AccessibilityService實(shí)現(xiàn)詳解
- Android?NotificationListenerService通知監(jiān)聽服務(wù)使用
- Android?NotificationListenerService?通知服務(wù)原理解析
- Android開發(fā)InputManagerService創(chuàng)建與啟動(dòng)流程
- Android 10 啟動(dòng)之servicemanager源碼解析
- Android 開機(jī)自啟動(dòng)Service實(shí)現(xiàn)詳解
相關(guān)文章
android studio 3.0 service項(xiàng)目背景音樂實(shí)現(xiàn)
這篇文章主要介紹了android studio 3.0中service項(xiàng)目實(shí)現(xiàn)插入背景音樂的方法。2017-11-11flutter PositionedTransition實(shí)現(xiàn)縮放動(dòng)畫
這篇文章主要為大家詳細(xì)介紹了flutter PositionedTransition實(shí)現(xiàn)縮放動(dòng)畫,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07Android使用WebView實(shí)現(xiàn)截圖分享功能
這篇文章主要為大家詳細(xì)介紹了Android使用WebView實(shí)現(xiàn)截圖分享功能,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-05-05Android 高德地圖之poi搜索功能的實(shí)現(xiàn)代碼
這篇文章主要介紹了android 高德地圖之poi搜索功能的實(shí)現(xiàn)代碼,在實(shí)現(xiàn)此功能時(shí)遇到很多問題,在文章都給大家提到,需要的朋友可以參考下2017-08-08Android中RecyclerView實(shí)現(xiàn)Item添加和刪除的代碼示例
本篇文章主要介紹了Android中RecyclerView實(shí)現(xiàn)Item添加和刪除的代碼示例,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-09-09Android實(shí)現(xiàn)實(shí)時(shí)通信示例
本篇文章主要介紹了Android實(shí)時(shí)通信示例,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-03-03Android學(xué)習(xí)之文件存儲(chǔ)讀取
本節(jié)給大家介紹的是Android數(shù)據(jù)存儲(chǔ)與訪問方式中的一個(gè)——文件存儲(chǔ)與讀寫,當(dāng)然除了這種方式外,我們可以存到SharedPreference,數(shù)據(jù)庫, 或者ContentProvider中,當(dāng)然這些后面都會(huì)講,嗯,開始本文內(nèi)容~2016-07-07