這兩天接手了一個(gè)需求氮采,就是盡量避免 App 被殺死,以保證推送的消息可以及時(shí)收到染苛。在接手前鹊漠,已經(jīng)有同事做了各種調(diào)研,最后決定使用前臺(tái)服務(wù)的方式以讓用戶(hù)對(duì) App 正在運(yùn)行的這種狀態(tài)有一個(gè)明確的感知「這里說(shuō)明下茶行,有些 App 會(huì)盡量避免讓用戶(hù)明確感知到前臺(tái)服務(wù)這種行為躯概,也會(huì)啟動(dòng)前臺(tái)服務(wù),不過(guò)會(huì)利用 Android 系統(tǒng)的 bug畔师,把這個(gè)前臺(tái)服務(wù)再隱藏娶靡,這個(gè)方式在網(wǎng)上有很多介紹」。
用startForeground(int id, Notification notification) 方式來(lái)啟動(dòng)前臺(tái)服務(wù)看锉,代碼并不多姿锭,在 AndroidManifest.xml 中再注冊(cè)一個(gè)服務(wù)塔鳍,在 Application 或者 Acitivity 啟動(dòng)即可,至于不同啟動(dòng)方式的利弊自己去查詢(xún)下呻此,我這個(gè)主要是看代碼寫(xiě)完后轮纫,各種主要機(jī)型的配合程度。
讓我吐槽一下焚鲜,這個(gè)過(guò)程我的心情是這樣的:懵『前臺(tái)服務(wù)掌唾?啟動(dòng)方式?代碼寫(xiě)好如何配合拉起推送的需求忿磅?』 --- 好像懂了『都知道了郑兴,就差拿手機(jī)試一試了』 --- 高興『用自己的 華為手機(jī)嘗試,success』--- 失望 or 絕望 『自己的 VIVO贝乎、公司的 OPPO均失敗情连,小米 success』--- 美滋滋『看到 VIVO、OPPO 起來(lái)了前臺(tái)服務(wù)览效,不過(guò)依然有難題... 』却舀。
前臺(tái)服務(wù)是否正常顯示最好的對(duì)比方式就是看在此手機(jī)上的高德地圖以及音樂(lè)播放器是否可以正常啟動(dòng)前臺(tái)服務(wù)。
來(lái)段啟動(dòng)前臺(tái)服務(wù)代碼:
<!--私信的前臺(tái)服務(wù)-->
<service android:name=".push.MsgService"/>
public class MsgService extends Service{
@Override
public void onCreate() {
super.onCreate();
Intent notificationIntent = new Intent(this, MsgChatHistoryActivity.class);
PendingIntent pendingIntent = PendingIntent.getActivity(this,0,notificationIntent,0);
Notification.Builder builder = new Notification.Builder(this.getApplicationContext());
builder.setContentTitle("***服務(wù)");
builder.setContentText("請(qǐng)勿關(guān)閉锤灿,***");
builder.setSmallIcon(R.drawable.gengmei_icon);
builder.setContentIntent(pendingIntent);
Notification notification = builder.getNotification();
startForeground(1,notification);//啟動(dòng)前臺(tái)服務(wù)
}
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
/**
* 當(dāng)Service因?yàn)閮?nèi)存不足而被系統(tǒng)kill后挽拔,接下來(lái)未來(lái)的某個(gè)時(shí)間內(nèi),當(dāng)系統(tǒng)內(nèi)存足夠可用的情況下但校,系統(tǒng)將會(huì)嘗試重新創(chuàng)建此Service螃诅,
* 一旦創(chuàng)建成功后將回調(diào)onStartCommand(...)方法,但其中的Intent將是null状囱,pendingintent除外
* 返回值可以確定下术裸,用哪個(gè)符合要求
*/
return START_STICKY;
}
@Nullable
@Override
public IBinder onBind(Intent intent) {
return null;
}
@Override
public void onDestroy() {
stopForeground(true);//停止前臺(tái)服務(wù)
super.onDestroy();
}
}
//啟動(dòng)服務(wù)
startService(new Intent(mContext,MsgService.class));
下面展示下這段代碼在幾種主流手機(jī)上的運(yùn)行效果:
小米手機(jī):
小米6、紅米note 4X 6.0亭枷、小米5.1.1 可以啟動(dòng)前臺(tái)服務(wù)袭艺,且不能撤銷(xiāo);「大概可以得出結(jié)論:小米手機(jī)可以正常啟動(dòng)前臺(tái)服務(wù)叨粘,且均不能撤銷(xiāo)猾编,除非自己去設(shè)置中單獨(dú)對(duì) App 進(jìn)行設(shè)置诗赌,但是一般的用戶(hù)不會(huì)有這種操作俏讹,可以先不用考慮」
三星手機(jī):
三星手機(jī) 5.0.1 系統(tǒng)可以正常啟動(dòng);
華為手機(jī):
華為6.0「默認(rèn)不會(huì)展開(kāi)『拒絕』『允許』的選擇框」幔嫂、4.4.2「默認(rèn)會(huì)展開(kāi)『拒絕』『允許』的選擇框」會(huì)有拒絕接受通知的情況驴党,上兩張圖說(shuō)明下瘪撇;「拒絕后,除非用戶(hù)手動(dòng)去設(shè)置中對(duì) App 單獨(dú)設(shè)置,否則不能再次開(kāi)啟设江,高德也是這樣的」
對(duì)比下小米,發(fā)現(xiàn)好喜歡小米攘轩,因?yàn)樾∶自谕ㄖ缑嬷懈揪蜎](méi)有「拒絕」「接受」的選擇框叉存,有效避免了用戶(hù)的這種「拒絕」的行為...
VIVO、OPPO手機(jī):
vivo X9Plus 笨篷、vivo X9 不能啟動(dòng)前臺(tái)服務(wù)瞳秽;「音樂(lè)播放器、高德可以啟動(dòng)前臺(tái)服務(wù)」
OPPO A59s 5.1版本 率翅、OPPO R9s 不能啟動(dòng)前臺(tái)服務(wù)练俐;「音樂(lè)播放器、高德地圖均不能啟動(dòng)前臺(tái)服務(wù)」
這兩種機(jī)型去網(wǎng)上查冕臭,Google 的結(jié)果簡(jiǎn)直就是讓人想哭腺晾,要么不能準(zhǔn)確定位問(wèn)題,要么不能準(zhǔn)確定位機(jī)型辜贵,更是讓我查到這樣的句子:vivo和oppo的手機(jī)上service基本無(wú)法常駐悯蝉,死的一干二凈明明白白,絲毫沒(méi)有任何掙扎的痕跡托慨,難道是我查的姿勢(shì)不對(duì)鼻由?「不過(guò)確實(shí)沒(méi)有做過(guò)單獨(dú)的某個(gè)機(jī)型的適配,不知道查某種機(jī)型的正確姿勢(shì)厚棵,待我找到后蕉世,再寫(xiě)一篇博客,哈哈...」
查了半天無(wú)果婆硬,OPPO 放到一邊不談讨彼,但是 VIVO 中 高德確實(shí)啟動(dòng)了前臺(tái)服務(wù),我開(kāi)始倒騰看手機(jī)到底有沒(méi)有相關(guān)的設(shè)置柿祈,難道真的像網(wǎng)上說(shuō)的OPPO等手機(jī)從根本上就拒絕了前臺(tái)服務(wù)這種行為哈误?姑娘并不想相信這個(gè)事實(shí)...
在 VIVO 中,高德安裝后默認(rèn)啟動(dòng)通知服務(wù)躏嚎,不需要手動(dòng)設(shè)置蜜自;但是,我目前的代碼沒(méi)能讓 VIVO 手機(jī)默認(rèn)啟動(dòng) App 通知服務(wù)卢佣,手動(dòng)設(shè)置后重荠,可以正常啟動(dòng)前臺(tái)服務(wù),所以下一步的任務(wù)就是找到讓 VIVO 手機(jī)默認(rèn)啟動(dòng) App 通知服務(wù)的正確姿勢(shì)虚茶,高德做到了所以一定有辦法...
在 OPPO 中戈鲁,高德安裝后默認(rèn)沒(méi)有啟動(dòng) App 通知服務(wù)仇参,需要手動(dòng)設(shè)置;我目前的代碼也沒(méi)能讓 OPPO 手機(jī)默認(rèn)啟動(dòng)通知服務(wù)婆殿,手動(dòng)設(shè)置后诈乒,倒是可以正常啟動(dòng)前臺(tái)服務(wù)。目前來(lái)看已經(jīng)打破了這個(gè)論斷:OPPO等手機(jī)上從根本上就拒絕了前臺(tái)服務(wù)這種行為婆芦。所以下一步的任務(wù)就是找到讓OPPO 手機(jī)默認(rèn)啟動(dòng) App 通知服務(wù)的正確姿勢(shì)怕磨,高德沒(méi)做到所以有點(diǎn)慫,不過(guò)可以試一試...
在 VIVO X9Plus手機(jī)上手動(dòng)設(shè)置的方式:
Settings -- More settings -- Applications -- All -- 目標(biāo) App -- Notifications -- 打開(kāi) Allow notification
在 OPPO R9s 手機(jī)上手動(dòng)設(shè)置的方式:
設(shè)置 -- 其他設(shè)置 -- 應(yīng)用程序管理 -- 已安裝 -- 目標(biāo) App --通知管理 -- 打開(kāi) 「允許通知」
再來(lái)一波鏈接:
前臺(tái)服務(wù)的使用
Service那點(diǎn)事兒
Android如何降低service被殺死概率
Android 如何開(kāi)啟前臺(tái)服務(wù)
GitHub 上的 Demo