日本免费高清视频-国产福利视频导航-黄色在线播放国产-天天操天天操天天操天天操|www.shdianci.com

學(xué)無(wú)先后,達(dá)者為師

網(wǎng)站首頁(yè) 編程語(yǔ)言 正文

Android?Doze模式下Alarm定時(shí)任務(wù)實(shí)現(xiàn)流程詳解_Android

作者:Tsuky1 ? 更新時(shí)間: 2022-12-06 編程語(yǔ)言

定時(shí)任務(wù)

Timer不太適合那些需要長(zhǎng)期在后臺(tái)運(yùn)行的定時(shí)任務(wù),因?yàn)槊總€(gè)手機(jī)都有自己的休眠策略,Android手機(jī)長(zhǎng)時(shí)間不操作就會(huì)導(dǎo)致Timer定時(shí)任務(wù)無(wú)法執(zhí)行,而Alarm具有喚醒CPU的功能,能保證大多數(shù)情況下,執(zhí)行定時(shí)任務(wù)的時(shí)候CPU能正常工作。

 AlarmManager manager= (AlarmManager) getSystemService(Context.ACTIVITY_SERVICE);
 PendingIntent pendingIntent;
      /**
         *    SystemClock.elapsedRealtime():獲取到系統(tǒng)開(kāi)機(jī)至今所經(jīng)歷的毫秒數(shù)
         *    System.currentTimeMillis():1970-1-1 0:00至今所經(jīng)歷的毫秒數(shù)
         *
         *     ELAPSED_REALTIME:定時(shí)任務(wù)觸發(fā)從系統(tǒng)開(kāi)機(jī)算起 但不喚醒CPU
         *     ELAPSED_REALTIME_WAKEUP:定時(shí)任務(wù)觸發(fā)從系統(tǒng)開(kāi)機(jī)算起 喚醒CPU
         *     RTC:從1970-1-1 0:00算起 不喚醒CPU
         *     RTC_WAKEUP:從1970-1-1 0:00算起 喚醒CPU
         */
    /**pendIntent:一般會(huì)調(diào)用getSrervice或getBroadCast
       來(lái)獲取一個(gè)能夠執(zhí)行服務(wù)或廣播的pendingIntent,
       這樣才能保證在任務(wù)被觸發(fā)的時(shí)候,
       服務(wù)里的onStartCommand()和onRecive()方法被執(zhí)行   
      */
   long time= SystemClock.elapsedRealtime()+10*1000;//10秒鐘后執(zhí)行任務(wù)
   manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,time,pendingIntent);

那么要實(shí)現(xiàn)一個(gè)長(zhǎng)時(shí)間在后臺(tái)定時(shí)運(yùn)行的服務(wù)該怎么做?

創(chuàng)建一個(gè)MyService類(lèi)

package com.example.test;
import android.app.AlarmManager;
import android.app.PendingIntent;
import android.app.Service;
import android.content.Intent;
import android.os.IBinder;
import android.os.SystemClock;
import androidx.annotation.Nullable;
public class MyService extends Service {
    @Nullable
    @Override
    public IBinder onBind(Intent intent) {
        return null;
    }
    @Override
    public int onStartCommand(Intent intent, int flags, int startId) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                //TODO
            }
        }).start();
        AlarmManager manager= (AlarmManager) getSystemService(ALARM_SERVICE);
        int time=60*60*1000;//一小時(shí)
        long triggerAtTime= SystemClock.elapsedRealtime()+time;
        Intent intent1=new Intent(this,MyService.class);
        PendingIntent pendingIntent=PendingIntent.getService(this,0,intent1,0);
        manager.set(AlarmManager.ELAPSED_REALTIME_WAKEUP,triggerAtTime,pendingIntent);
        return super.onStartCommand(intent, flags, startId);
    }
}

可以看到先創(chuàng)建了一個(gè)子線程在這里執(zhí)行具體的操作任務(wù),然后就是alarm機(jī)制,定義任務(wù)觸發(fā)一小時(shí)后,再使用PendingIntent指定定時(shí)處理任務(wù)的MyService,最后調(diào)用set方法設(shè)定完成。

這樣就將一個(gè)長(zhǎng)時(shí)間在后臺(tái)定時(shí)運(yùn)行的服務(wù)設(shè)定成功了,一旦啟動(dòng)了MyService,就會(huì)在onStartCommand()方法里設(shè)定一個(gè)定時(shí)任務(wù),這樣一小時(shí)后將會(huì)再次啟動(dòng)MyService,每隔一小時(shí)就會(huì)執(zhí)行一次。

然后去使用,在manifest文件中注冊(cè)服務(wù)

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

在使用的地方調(diào)用下面代碼就可以。

    Intent intent=new Intent(context,MyService.class);
    context.startService(intent);

但是在Android4.4后,定時(shí)任務(wù)變得不準(zhǔn)確,會(huì)延時(shí)一段時(shí)間才能執(zhí)行,那是因?yàn)橄到y(tǒng)在耗電方面進(jìn)行的優(yōu)化,系統(tǒng)會(huì)檢測(cè)有多少Alarm任務(wù)存在,將觸發(fā)時(shí)間相近的幾個(gè)任務(wù)放在一起執(zhí)行,這樣可以大幅度的減少CPU被喚醒的次數(shù),演唱電池的使用時(shí)長(zhǎng)。也可以使用setExact()來(lái)代替set()方法,可以保證任務(wù)能夠準(zhǔn)時(shí)執(zhí)行。

Doze模式

如果設(shè)備未充電,并處于靜止?fàn)顟B(tài),且屏幕關(guān)閉了一段時(shí)間后,就能進(jìn)入到Doze模式。

在Doze模式下,系統(tǒng)會(huì)對(duì)CPU、網(wǎng)絡(luò)、Alarm等活動(dòng)進(jìn)行限制,從而延長(zhǎng)電池的使用壽命。當(dāng)然,系統(tǒng)并不會(huì)一直處于Doze模式,而是會(huì)間歇性地退出Doze模式一小段時(shí)間,在這段時(shí)間中,應(yīng)用就可以去完成它的同步操作、Alarm任務(wù)等。

Doze模式下受限制的功能:

  • 網(wǎng)絡(luò)訪問(wèn)被禁止。
  • 系統(tǒng)忽略喚醒CPU或者屏幕操作。
  • 系統(tǒng)不再執(zhí)行WIFI掃描。
  • 系統(tǒng)不再執(zhí)行同步服務(wù)。
  • Alarm任務(wù)將會(huì)在下次退出Doze模式的時(shí)候執(zhí)行。

在Doze模式下, Alarm任務(wù)會(huì)變得相當(dāng)不準(zhǔn),當(dāng)然這種情況下對(duì)Alarm的要求也并不高,如果有特殊的要求,調(diào)用AlarmManager的setAndAllowWhileIdle()或者setExactAndAllowWhileIdle()方法就能讓定時(shí)任務(wù)正常執(zhí)行,和set()與setExact()的區(qū)別是一樣的。

原文鏈接:https://blog.csdn.net/m0_56366502/article/details/127559760

欄目分類(lèi)
最近更新