1.概述
在Android系統中,鬧鐘和喚醒功能都是由Alarm Manager Service控制並管理的。咱們所熟悉的RTC鬧鐘以及定時器都和它有莫大的關係。爲了便於稱呼,我經常也把這個service簡稱爲ALMS。java
另外,ALMS還提供了一個AlarmManager輔助類。在實際的代碼中,應用程序通常都是經過這個輔助類來和ALMS打交道的。就代碼而言,輔助類只不過是把一些邏輯語義傳遞給ALMS服務端而已,具體怎麼作則徹底要看ALMS的實現代碼了。android
ALMS的實現代碼並不算太複雜,主要只是在管理「邏輯鬧鐘」。它把邏輯鬧鐘分紅幾個大類,分別記錄在不一樣的列表中。而後ALMS會在一個專門的線程中循環等待鬧鐘的激發,一旦時機到了,就「回調」邏輯鬧鐘對應的動做。數組
以上只是一些概要性的介紹,下面咱們來看具體的技術細節。安全
先看下具體ALMS在應用中的使用app
- 1. Intent intent = new Intent(this, OneShotAlarm.class);
- 2. PendingIntent sender = PendingIntent.getBroadcast(this, 0, intent, 0);
- 3.
- 4. // 設置警報時間
- 5. Calendar calendar = Calendar.getInstance();
- 6. calendar.setTimeInMillis(System.currentTimeMillis());
- 7. calendar.add(Calendar.SECOND, 30);
- 8.
- 9. // 設置警報時間,除了用Calendar以外,還能夠用
- 10. long firstTime = SystemClock.elapsedRealtime();
- 11.
- 12. AlarmManager am = (AlarmManager) getSystemService(ALARM_SERVICE);
- 13. // 只會警報一次
- 14. am.set(AlarmManager.RTC_WAKEUP, calendar.getTimeInMillis(), sender);
- 15. // 會重複警報屢次
- 16. am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstTime, 15*1000, sender);
- 17.
- 18. // 要取消這個警報,只要經過PendingIntent就能夠作到
- 19. am.cancel(sender);
2.AlarmManager
前文咱們已經說過,ALMS只是服務端的東西。它必須向外提供具體的接口,才能被外界使用。在android平臺中,ALMS的外部接口爲IAlarmManager。其定義位於frameworks\base\core\Java\android\app\IAlarmManager.aidl腳本中,定義截選以下: 框架
interface IAlarmManager {
void set(int type, long triggerAtTime, in PendingIntent operation);
void setRepeating(int type, long triggerAtTime, long interval, in PendingIntent operation);
void setInexactRepeating(int type, long triggerAtTime, long interval, in PendingIntent operation);
void setTime(long millis);
void setTimeZone(String zone);
void remove(in PendingIntent operation);
}
在通常狀況下,service的使用者會經過Service Manager Service接口,先拿到它感興趣的service對應的代理I接口,而後再調用I接口的成員函數向service發出請求。因此按理說,咱們也應該先拿到一個IAlarmManager接口,而後再使用它。但是,對Alarm Manager Service來講,狀況略有不一樣,其最多見的調用方式以下:less
manager = (AlarmManager)context.getSystemService(Context.ALARM_SERVICE);
其中,getSystemService()返回的再也不是IAlarmManager接口,而是AlarmManager對象。 ide
2.1 AlarmManager的成員函數
AlarmManager的成員函數有:函數
AlarmManager(IAlarmManager service)
publicvoid set(int type,long triggerAtTime, PendingIntent operation)
publicvoid setRepeating(int type,long triggerAtTime,long interval,
PendingIntent operation)
publicvoid setInexactRepeating(int type,long triggerAtTime,long interval,
PendingIntent operation)
publicvoid cancel(PendingIntent operation)
publicvoid setTime(long millis)
publicvoid setTimeZone(String timeZone)即1個構造函數,6個功能函數。基本上徹底和IAlarmManager的成員函數一一對應。 ui
另外,AlarmManager類中會以不一樣的公共常量來表示多種不一樣的邏輯鬧鐘,在Android 4.0的原生代碼中有4種邏輯鬧鐘:
1) RTC_WAKEUP
2) RTC
3) ELAPSED_REALTIME_WAKEUP
4) ELAPSED_REALTIME
應用側經過調用AlarmManager對象的成員函數,能夠把語義傳遞到AlarmManagerService,並由它進行實際的處理。
3.AlarmManagerService
ALMS的重頭戲在AlarmManagerService中,這個類繼承於IAlarmManager.Stub,因此是個binder實體。它包含的重要成員以下:
其中,mRtcWakeupAlarms等4個ArrayList<Alarm>數組分別對應着前文所說的4種「邏輯鬧鐘」。爲了便於理解,咱們能夠想象在底層有4個「實體鬧鐘」,注意,是4個,不是4類。上面每一類「邏輯鬧鐘」都會對應一個「實體鬧鐘」,而邏輯鬧鐘則能夠有若干個,它們被存儲在ArrayList中,示意圖以下:
固然,這裏所說的「實體鬧鐘」只是個概念而已,其具體實現和底層驅動有關,在frameworks層沒必要過多關心。
Frameworks層應該關心的是那幾個ArrayList<Alarm>。這裏的Alarm對應着邏輯鬧鐘。
3.1 邏輯鬧鐘
Alarm是AlarmManagerService的一個內嵌類Alarm,定義截選以下:
- private static class Alarm {
- public int type;
- public int count;
- public long when;
- public long repeatInterval;
- public PendingIntent operation;
- public int uid;
- public int pid;
- . . . . . .
其中記錄了邏輯鬧鐘的一些關鍵信息。
- type域:記錄着邏輯鬧鐘的鬧鐘類型,好比RTC_WAKEUP、ELAPSED_REALTIME_WAKEUP等;
- count域:是個輔助域,它和repeatInterval域一塊兒工做。當repeatInterval大於0時,這個域可被用於計算下一次重複激發alarm的時間,詳細狀況見後文;
- when域:記錄鬧鐘的激發時間。這個域和type域相關,詳細狀況見後文;
- repeatInterval域:表示重複激發鬧鐘的時間間隔,若是鬧鐘只需激發一次,則此域爲0,若是鬧鐘須要重複激發,此域爲以毫秒爲單位的時間間隔;
- operation域:記錄鬧鐘激發時應該執行的動做,詳細狀況見後文;
- uid域:記錄設置鬧鐘的進程的uid;
- pid域:記錄設置鬧鐘的進程的pid。
整體來講仍是比較簡單的,咱們先補充說明一下其中的count域。這個域是針對重複性鬧鐘的一個輔助域。重複性鬧鐘的實現機理是,若是當前時刻已經超過鬧鐘的激發時刻,那麼ALMS會先從邏輯鬧鐘數組中摘取下Alarm節點,並執行鬧鐘對應的邏輯動做,而後進一步比較「當前時刻」和「Alarm理應激發的理想時刻」之間的時間跨度,從而計算出Alarm的「下一次理應激發的理想時刻」,並將這個激發時間記入Alarm節點,接着將該節點從新排入邏輯鬧鐘列表。這一點和普通Alarm不太同樣,普通Alarm節點摘下後就再也不還回邏輯鬧鐘列表了。
「當前時刻」和「理應激發時刻」之間的時間跨度會隨實際的運做狀況而變更。咱們分兩步來講明「下一次理應激發時刻」的計算公式:
1) count = (時間跨度 / repeatInterval ) + 1 ;
2) 「下一次理應激發時刻」 = 「上一次理應激發時刻」+ count * repeatInterval ;
咱們畫一張示意圖,其中綠色的可激發時刻表示「上一次理應激發時刻」,咱們假定「當前時刻」分別爲now_1處或now_2處,能夠看到會計算出不一樣的「下一次理應激發時刻」,這裏用桔紅色表示。
能夠看到,若是當前時刻爲now_1,那麼它和「上一次理應激發時刻」之間的「時間跨度」是小於一個repeatInterval的,因此count數爲1。而若是當前時刻爲now_2,那麼「時間跨度」與repeatInterval的商取整後爲2,因此count數爲3。另外,圖中那兩個虛線箭頭對應的可激發時刻,只是用來作刻度的東西。
3.2 主要行爲
接下來咱們來看ALMS中的主要行爲,這些行爲和AlarmManager輔助類提供的成員函數相對應。
3.2.1 設置alarm
外界能接觸的設置alarm的函數是set():publicvoid set(int type,long triggerAtTime, PendingIntent operation)
type:表示要設置的alarm類型。如前文所述,有4個alarm類型。
triggerAtTime:表示alarm「理應激發」的時間。
operation:指明瞭alarm鬧鈴激發時須要執行的動做,好比執行某種廣播通告。
設置alarm的動做會牽扯到一個發起者。簡單地說,發起者會向Alarm Manager Service發出一個設置alarm的請求,並且在請求裏註明了到時間後須要執行的動做。因爲「待執行的動做」通常都不會立刻執行,因此要表達成PendingIntent的形式。(PendingIntent的詳情可參考其餘文章)
另外,triggerAtTime參數的意義也會隨type參數的不一樣而不一樣。簡單地說,若是type是和RTC相關的話,那麼triggerAtTime的值應該是標準時間,即從1970 年 1 月 1 日午夜開始所通過的毫秒數。而若是type是其餘類型的話,那麼triggerAtTime的值應該是從本次開機開始算起的毫秒數。
3.2.2 重複性alarm
另外一個設置alarm的函數是setRepeating():
- public void setRepeating(int type, long triggerAtTime, long interval,PendingIntent operation)
其參數基本上和set()函數差很少,只是多了一個「時間間隔」參數。事實上,在Alarm Manager Service一側,set()函數內部也是在調用setRepeating()的,只不過會把interval設成了0。
setRepeating()的實現函數以下:
- public void setRepeating(int type, long triggerAtTime, long interval,
- PendingIntent operation)
- {
- if (operation == null) {
- Slog.w(TAG, "set/setRepeating ignored because there is no intent");
- return;
- }
- synchronized (mLock) {
- Alarm alarm = new Alarm();
- alarm.type = type;
- alarm.when = triggerAtTime;
- alarm.repeatInterval = interval;
- alarm.operation = operation;
-
- // Remove this alarm if already scheduled.
- removeLocked(operation);
-
- if (localLOGV) Slog.v(TAG, "set: " + alarm);
-
- int index = addAlarmLocked(alarm);
- if (index == 0) {
- setLocked(alarm);
- }
- }
- }
代碼很簡單,會建立一個邏輯鬧鐘Alarm,然後調用addAlarmLocked()將邏輯鬧鐘添加到內部邏輯鬧鐘數組的某個合適位置。
- private int addAlarmLocked(Alarm alarm) {
- ArrayList<Alarm> alarmList = getAlarmList(alarm.type);
-
- int index = Collections.binarySearch(alarmList, alarm, mIncreasingTimeOrder);
- if (index < 0) {
- index = 0 - index - 1;
- }
- if (localLOGV) Slog.v(TAG, "Adding alarm " + alarm + " at " + index);
- alarmList.add(index, alarm);
- . . . . . .
- return index;
- }
邏輯鬧鐘列表是依據alarm的激發時間進行排序的,越早激發的alarm,越靠近第0位。因此,addAlarmLocked()在添加新邏輯鬧鐘時,須要先用二分查找法快速找到列表中合適的位置,而後再把Alarm對象插入此處。
若是所插入的位置正好是第0位,就說明此時新插入的這個邏輯鬧鐘將會是本類alarm中最早被激發的alarm,而正如咱們前文所述,每一類邏輯鬧鐘會對應同一個「實體鬧鐘」,此處咱們在第0位設置了新的激發時間,明確表示咱們之前對「實體鬧鐘」設置的激發時間已經不許確了,因此setRepeating()中必須從新調整一下「實體鬧鐘」的激發時間,因而有了下面的句子:
- if (index == 0) {
- setLocked(alarm);
- }
setLocked()內部會調用native函數set():
- private native void set(int fd, int type, long seconds, long nanoseconds);
從新設置「實體鬧鐘」的激發時間。這個函數內部會調用ioctl()和底層打交道。具體代碼可參考frameworks/base/services/jni/com_android_server_AlarmManagerService.cpp文件:
- static void android_server_AlarmManagerService_set(JNIEnv* env, jobject obj, jint fd,
- jint type, jlong seconds, jlong nanoseconds)
- {
- struct timespec ts;
- ts.tv_sec = seconds;
- ts.tv_nsec = nanoseconds;
-
- int result = ioctl(fd, ANDROID_ALARM_SET(type), &ts);
- if (result < 0)
- {
- ALOGE("Unable to set alarm to %lld.%09lld: %s\n", seconds, nanoseconds, strerror(errno));
- }
- }
咱們知道,PendingIntent只是frameworks一層的概念,和底層驅動是沒有關係的。因此向底層設置alarm時只須要type信息以及激發時間信息就能夠了。
在AlarmManagerService中真正設置alarm的函數是setImplLocked函數,在這個函數中把alarm添加到mAlarmBatchs中,mAlarmBatchs會把觸發時間相近的Alarm放在同一個bach中,而後每一個bach根據時間排序放在mAlarmBatchs中,前面的就是先要觸發的alarm。
- private void setImplLocked(int type, long when, long whenElapsed, long maxWhen, long interval,
- PendingIntent operation, boolean isStandalone, boolean doValidate,
- WorkSource workSource) {
- /**建立一個alarm,其中各參數的含義以下:
- * type 鬧鐘類型 ELAPSED_REALTIME、RTC、RTC_WAKEUP等
- * when 觸發時間 UTC類型,絕對時間,經過System.currentTimeMillis()獲得
- * whenElapsed 相對觸發時間,自開機算起,含休眠,經過SystemClock.elapsedRealtime()獲得
- * maxWhen 最大觸發時間
- * interval 觸發間隔,針對循環鬧鐘有效
- * operation 鬧鐘觸發時的行爲,PendingIntent類型
- */
- Alarm a = new Alarm(type, when, whenElapsed, maxWhen, interval, operation, workSource);
- //根據PendingIntent刪除以前已有的同一個鬧鐘
- removeLocked(operation);
-
- boolean reschedule;
- //嘗試將alarm加入到合適的batch中,若是alarm是獨立的或者沒法找到合適的batch去容納此alarm,返回-1
- int whichBatch = (isStandalone) ? -1 : attemptCoalesceLocked(whenElapsed, maxWhen);
- if (whichBatch < 0) {
- //沒有合適的batch去容納alarm,則新建一個batch
- Batch batch = new Batch(a);
- batch.standalone = isStandalone;
- //將batch加入mAlarmBatches中,並對mAlarmBatches進行排序:按開始時間升序排列
- reschedule = addBatchLocked(mAlarmBatches, batch);
- } else {
- //若是找到合適了batch去容納此alarm,則將其加入到batch中
- Batch batch = mAlarmBatches.get(whichBatch);
- //若是當前alarm的加入引發了batch開始時間和結束時間的改變,則reschedule爲true
- reschedule = batch.add(a);
- if (reschedule) {
- //因爲batch的起始時間發生了改變,因此須要從列表中刪除此batch並從新加入、從新對batch列表進行排序
- mAlarmBatches.remove(whichBatch);
- addBatchLocked(mAlarmBatches, batch);
- }
- }
-
- if (DEBUG_VALIDATE) {
- if (doValidate && !validateConsistencyLocked()) {
- Slog.v(TAG, "Tipping-point operation: type=" + type + " when=" + when
- + " when(hex)=" + Long.toHexString(when)
- + " whenElapsed=" + whenElapsed + " maxWhen=" + maxWhen
- + " interval=" + interval + " op=" + operation
- + " standalone=" + isStandalone);
- rebatchAllAlarmsLocked(false);
- reschedule = true;
- }
- }
-
- if (reschedule) {
- rescheduleKernelAlarmsLocked();
- }
- }
- rescheduleKernelAlarmsLocked函數主要用來選取alarm的觸發時間設置到RTC中去。
- private void rescheduleKernelAlarmsLocked() {
- // Schedule the next upcoming wakeup alarm. If there is a deliverable batch
- // prior to that which contains no wakeups, we schedule that as well.
- if (mAlarmBatches.size() > 0) {
- //查找第一個有wakeup類型alarm的batch
- final Batch firstWakeup = findFirstWakeupBatchLocked();
- //查找第一個batch
- final Batch firstBatch = mAlarmBatches.get(0);
- 判斷條件是爲了防止重複設置
- if (firstWakeup != null && mNextWakeup != firstWakeup.start) {
- //將第一個有wakeup類型alarm的batch的時間設置到rtc中
- mNextWakeup = firstWakeup.start;
- setLocked(ELAPSED_REALTIME_WAKEUP, firstWakeup.start);
- }
- if (firstBatch != firstWakeup && mNextNonWakeup != firstBatch.start) {
- mNextNonWakeup = firstBatch.start;
- setLocked(ELAPSED_REALTIME, firstBatch.start);
- }
- }
- }
3.2.3 取消alarm
用戶端是調用AlarmManager對象的cancel()函數來取消alarm的。這個函數內部實際上是調用IAlarmManager的remove()函數。因此咱們只來看AlarmManagerService的remove()就能夠了。
- public void remove(PendingIntent operation)
- {
- if (operation == null) {
- return;
- }
- synchronized (mLock) {
- removeLocked(operation);
- }
- }
注意,在取消alarm時,是以一個PendingIntent對象做爲參數的。這個PendingIntent對象正是當初設置alarm時,所傳入的那個operation參數。咱們不能隨便建立一個新的PendingIntent對象來調用remove()函數,不然remove()是不會起做用的。PendingIntent的運做細節不在本文論述範圍以內,此處咱們只需粗淺地知道,PendingIntent對象在AMS(Activity Manager Service)端會對應一個PendingIntentRecord實體,而ALMS在遍歷邏輯鬧鐘列表時,是根據是否指代相同PendingIntentRecord實體來判斷PendingIntent的相符狀況的。若是咱們隨便建立一個PendingIntent對象並傳入remove()函數的話,那麼在ALMS端勢必找不到相符的PendingIntent對象,因此remove()必然無效。
remove()中調用的removeLocked()以下:
- public void removeLocked(PendingIntent operation)
- {
- removeLocked(mRtcWakeupAlarms, operation);
- removeLocked(mRtcAlarms, operation);
- removeLocked(mElapsedRealtimeWakeupAlarms, operation);
- removeLocked(mElapsedRealtimeAlarms, operation);
- }
簡單地說就是,把4個邏輯鬧鐘數組都遍歷一遍,刪除其中全部和operation相符的Alarm節點。removeLocked()的實現代碼以下:
- private void removeLocked(ArrayList<Alarm> alarmList,
- PendingIntent operation)
- {
- if (alarmList.size() <= 0) {
- return;
- }
-
- // iterator over the list removing any it where the intent match
- Iterator<Alarm> it = alarmList.iterator();
-
- while (it.hasNext()) {
- Alarm alarm = it.next();
- if (alarm.operation.equals(operation)) {
- it.remove();
- }
- }
- }
請注意,所謂的取消alarm,只是刪除了對應的邏輯Alarm節點而已,並不會和底層驅動再打什麼交道。也就是說,是不存在針對底層「實體鬧鐘」的刪除動做的。因此,底層「實體鬧鐘」在到時之時,仍是會被「激發」出來的,只不過此時在frameworks層,會由於找不到符合要求的「邏輯鬧鐘」而不作進一步的激發動做。
3.2.4 設置系統時間和時區
AlarmManager還提供設置系統時間的功能,設置者須要具備android.permission.SET_TIME權限。
- public void setTime(long millis)
- {
- mContext.enforceCallingOrSelfPermission("android.permission.SET_TIME", "setTime");
- SystemClock.setCurrentTimeMillis(millis);
- }
另外,還具備設置時區的功能:
- public void setTimeZone(String tz)
相應地,設置者須要具備android.permission.SET_TIME_ZONE權限。
3.3 運做細節
3.3.1 AlarmThread和Alarm的激發
AlarmManagerService內部是如何感知底層激發alarm的呢?首先,AlarmManagerService有一個表示線程的mWaitThread成員:
- private final AlarmThread mWaitThread = new AlarmThread();
在AlarmManagerService構造之初,就會啓動這個專門的「等待線程」。
- public AlarmManagerService(Context context)
- {
- mContext = context;
- mDescriptor = init();
- . . . . . .
- . . . . . .
- if (mDescriptor != -1)
- {
- mWaitThread.start(); // 啓動線程!
- }
- else
- {
- Slog.w(TAG, "Failed to open alarm driver. Falling back to a handler.");
- }
- }
AlarmManagerService的構造函數一開始就會調用一個init()函數,該函數是個native函數,它的內部會打開alarm驅動,並返回驅動文件句柄。只要可以順利打開alarm驅動,ALMS就能夠走到mWaitThread.start()一句,因而「等待線程」就啓動了。
3.3.1.1 AlarmThread中的run()
AlarmThread自己是AlarmManagerService中一個繼承於Thread的內嵌類:
- private class AlarmThread extends Thread
其最核心的run()函數的主要動做流程圖以下:
咱們分別來闡述上圖中的關鍵步驟。
3.3.1.2 waitForAlarm()
首先,從上文的流程圖中能夠看到,AlarmThread線程是在一個while(true)循環裏不斷調用waitForAlarm()函數來等待底層alarm激發動做的。waitForAlarm()是一個native函數:
- private native int waitForAlarm(int fd);
其對應的C++層函數是android_server_AlarmManagerService_waitForAlarm():
【com_android_server_AlarmManagerService.cpp】
- static jint android_server_AlarmManagerService_waitForAlarm(JNIEnv* env, jobject obj, jint fd)
- {
- int result = 0;
-
- do
- {
- result = ioctl(fd, ANDROID_ALARM_WAIT);
- } while (result < 0 && errno == EINTR);
-
- if (result < 0)
- {
- ALOGE("Unable to wait on alarm: %s\n", strerror(errno));
- return 0;
- }
-
- return result;
- }
當AlarmThread調用到ioctl()一句時,線程會阻塞住,直到底層激發alarm。並且所激發的alarm的類型會記錄到ioctl()的返回值中。這個返回值對外界來講很是重要,外界用它來判斷該遍歷哪一個邏輯鬧鐘列表。
3.3.1.3 triggerAlarmsLocked()
一旦等到底層驅動的激發動做,AlarmThread會開始遍歷相應的邏輯鬧鐘列表:
- ArrayList<Alarm> triggerList = new ArrayList<Alarm>();
- . . . . . .
- final long nowRTC = System.currentTimeMillis();
- final long nowELAPSED = SystemClock.elapsedRealtime();
- . . . . . .
- if ((result & RTC_WAKEUP_MASK) != 0)
- triggerAlarmsLocked(mRtcWakeupAlarms, triggerList, nowRTC);
- if ((result & RTC_MASK) != 0)
- triggerAlarmsLocked(mRtcAlarms, triggerList, nowRTC);
- if ((result & ELAPSED_REALTIME_WAKEUP_MASK) != 0)
- triggerAlarmsLocked(mElapsedRealtimeWakeupAlarms, triggerList, nowELAPSED);
- if ((result & ELAPSED_REALTIME_MASK) != 0)
- triggerAlarmsLocked(mElapsedRealtimeAlarms, triggerList, nowELAPSED);
能夠看到,AlarmThread先建立了一個臨時的數組列表triggerList,而後根據result的值對相應的alarm數組列表調用triggerAlarmsLocked(),一旦發現alarm數組列表中有某個alarm符合激發條件,就把它移到triggerList中。這樣,4條alarm數組列表中須要激發的alarm就彙總到triggerList數組列表中了。
triggerAlarmsLocked函數主要將要發送的alarm降入triggerlist中
- private void triggerAlarmsLocked(ArrayList<Alarm> triggerList, long nowELAPSED, long nowRTC) {
- // batches are temporally sorted, so we need only pull from the
- // start of the list until we either empty it or hit a batch
- // that is not yet deliverable
- while (mAlarmBatches.size() > 0) {
- //獲取第一個batch
- Batch batch = mAlarmBatches.get(0);
- if (batch.start > nowELAPSED) {
- // Everything else is scheduled for the future
- break;
- }
-
- // We will (re)schedule some alarms now; don't let that interfere
- // with delivery of this current batch
- //將第一個batch去除
- mAlarmBatches.remove(0);
-
- final int N = batch.size();
- for (int i = 0; i < N; i++) {
- Alarm alarm = batch.get(i);
- alarm.count = 1;
- //遍歷加入triggerList
- triggerList.add(alarm);
-
- // Recurring alarms may have passed several alarm intervals while the
- // phone was asleep or off, so pass a trigger count when sending them.
- //若是有重複類型的,計算時間從新設置
- if (alarm.repeatInterval > 0) {
- // this adjustment will be zero if we're late by
- // less than one full repeat interval
- alarm.count += (nowELAPSED - alarm.whenElapsed) / alarm.repeatInterval;
-
- // Also schedule its next recurrence
- final long delta = alarm.count * alarm.repeatInterval;
- final long nextElapsed = alarm.whenElapsed + delta;
- setImplLocked(alarm.type, alarm.when + delta, nextElapsed, alarm.windowLength,
- maxTriggerTime(nowELAPSED, nextElapsed, alarm.repeatInterval),
- alarm.repeatInterval, alarm.operation, batch.standalone, true,
- alarm.workSource);
- }
-
- }
- }
- }
接下來,只需遍歷一遍triggerList就能夠了:
- Iterator<Alarm> it = triggerList.iterator();
- while (it.hasNext())
- {
- Alarm alarm = it.next();
- . . . . . .
- alarm.operation.send(mContext, 0,
- mBackgroundIntent.putExtra(Intent.EXTRA_ALARM_COUNT, alarm.count),
- mResultReceiver, mHandler);
-
- // we have an active broadcast so stay awake.
- if (mBroadcastRefCount == 0) {
- setWakelockWorkSource(alarm.operation);
- mWakeLock.acquire();
- }
- mInFlight.add(alarm.operation);
- mBroadcastRefCount++;
- mTriggeredUids.add(new Integer(alarm.uid));
- BroadcastStats bs = getStatsLocked(alarm.operation);
- if (bs.nesting == 0) {
- bs.startTime = nowELAPSED;
- } else {
- bs.nesting++;
- }
- if (alarm.type == AlarmManager.ELAPSED_REALTIME_WAKEUP
- || alarm.type == AlarmManager.RTC_WAKEUP) {
- bs.numWakeup++;
- ActivityManagerNative.noteWakeupAlarm(alarm.operation);
- }
- }
在上面的while循環中,每遍歷到一個Alarm對象,就執行它的alarm.operation.send()函數。咱們知道,alarm中記錄的operation就是當初設置它時傳來的那個PendingIntent對象,如今開始執行PendingIntent的send()操做啦。
PendingIntent的send()函數代碼是:
- public void send(Context context, int code, Intent intent,
- OnFinished onFinished, Handler handler) throws CanceledException
- {
- send(context, code, intent, onFinished, handler, null);
- }
-
調用了下面的send()函數:
- public void send(Context context, int code, Intent intent,
- OnFinished onFinished, Handler handler, String requiredPermission)
- throws CanceledException
- {
- try
- {
- String resolvedType = intent != null
- ? intent.resolveTypeIfNeeded(context.getContentResolver())
- : null;
- int res = mTarget.send(code, intent, resolvedType,
- onFinished != null
- ? new FinishedDispatcher(this, onFinished, handler)
- : null,
- requiredPermission);
- if (res < 0)
- {
- throw new CanceledException();
- }
- }
- catch (RemoteException e)
- {
- throw new CanceledException(e);
- }
- }
mTarget是個IPendingIntent代理接口,它對應AMS(Activity Manager Service)中的某個PendingIntentRecord實體。須要說明的是,PendingIntent的重要信息都是在AMS的PendingIntentRecord以及PendingIntentRecord.Key對象中管理的。AMS中有一張哈希表專門用於記錄全部可用的PendingIntentRecord對象。
相較起來,在建立PendingIntent對象時傳入的intent數組,其重要性並不太明顯。這種intent數組主要用於一次性啓動多個activity,若是你只是但願啓動一個activity或一個service,那麼這個intent的內容有可能在最終執行PendingIntent的send()動做時,被新傳入的intent內容替換掉。
AMS中關於PendingIntentRecord哈希表的示意圖以下:
AMS是整個Android平臺中最複雜的一個核心service了,因此咱們不在這裏作過多的闡述,有興趣的讀者能夠參考其餘相關文檔。
3.3.1.4 進一步處理「喚醒鬧鐘」
在AlarmThread.run()函數中while循環的最後,會進一步判斷,當前激發的alarm是否是「喚醒鬧鐘」。若是鬧鐘類型爲RTC_WAKEUP或ELAPSED_REALTIME_WAKEUP,那它就屬於「喚醒鬧鐘」,此時須要通知一下AMS:
- if (alarm.type == AlarmManager.ELAPSED_REALTIME_WAKEUP
- || alarm.type == AlarmManager.RTC_WAKEUP)
- {
- bs.numWakeup++;
- ActivityManagerNative.noteWakeupAlarm(alarm.operation);
- }
這兩種alarm就是咱們常說的0型和2型鬧鐘,它們和咱們手機的續航時間息息相關。
AMS裏的noteWakeupAlarm()比較簡單,只是在調用BatteryStatsService服務的相關動做,可是卻會致使機器的喚醒:
- public void noteWakeupAlarm(IIntentSender sender)
- {
- if (!(sender instanceof PendingIntentRecord))
- {
- return;
- }
-
- BatteryStatsImpl stats = mBatteryStatsService.getActiveStatistics();
- synchronized (stats)
- {
- if (mBatteryStatsService.isOnBattery())
- {
- mBatteryStatsService.enforceCallingPermission();
- PendingIntentRecord rec = (PendingIntentRecord)sender;
- int MY_UID = Binder.getCallingUid();
- int uid = rec.uid == MY_UID ? Process.SYSTEM_UID : rec.uid;
- BatteryStatsImpl.Uid.Pkg pkg = stats.getPackageStatsLocked(uid, rec.key.packageName);
- pkg.incWakeupsLocked();
- }
- }
- }
好了,說了這麼多,咱們仍是畫一張AlarmThread示意圖做爲總結:
3.3.2 說說AlarmManagerService中的mBroadcastRefCount
下面咱們說說AlarmManagerService中的mBroadcastRefCount,之因此要說它,僅僅是由於我在修改AlarmManagerService代碼的時候,吃過它的虧。
咱們先回顧一下處理triggerList列表的代碼,以下:
- Iterator<Alarm> it = triggerList.iterator();
- while (it.hasNext())
- {
- Alarm alarm = it.next();
- . . . . . .
- alarm.operation.send(mContext, 0,
- mBackgroundIntent.putExtra(Intent.EXTRA_ALARM_COUNT, alarm.count),
- mResultReceiver, mHandler);
-
- // we have an active broadcast so stay awake.
- if (mBroadcastRefCount == 0) {
- setWakelockWorkSource(alarm.operation);
- mWakeLock.acquire();
- }
- mInFlight.add(alarm.operation);
- mBroadcastRefCount++;
- . . . . . .
- . . . . . .
- }
能夠看到,在AlarmThread.run()中,只要triggerList中含有可激發的alarm,mBroadcastRefCount就會執行加一操做。一開始mBroadcastRefCount的值爲0,因此會進入上面那句if語句,進而調用mWakeLock.acquire()。
後來我才知道,這個mBroadcastRefCount變量,是決定什麼時候釋放mWakeLock的計數器。AlarmThread的意思很明確,只要還有處於激發狀態的邏輯鬧鐘,機器就不能徹底睡眠。那麼釋放這個mWakeLock的地方又在哪裏呢?答案就在alarm.operation.send()一句的mResultReceiver參數中。
mResultReceiver是AlarmManagerService的私有成員變量:
- private final ResultReceiver mResultReceiver = newResultReceiver();
類型爲ResultReceiver,這個類實現了PendingIntent.OnFinished接口:
- class ResultReceiver implements PendingIntent.OnFinished
當send()動做完成後,框架會間接回調這個對象的onSendFinished()成員函數。
- public void onSendFinished(PendingIntent pi, Intent intent, int resultCode,
- String resultData, Bundle resultExtras)
- {
- . . . . . .
- . . . . . .
- if (mBlockedUids.contains(new Integer(uid)))
- {
- mBlockedUids.remove(new Integer(uid));
- }
- else
- {
- if (mBroadcastRefCount > 0)
- {
- mInFlight.removeFirst();
- mBroadcastRefCount--;
-
- if (mBroadcastRefCount == 0)
- {
- mWakeLock.release();
- }
- . . . . . .
- }
- . . . . . .
- }
- . . . . . .
- }
我一開始沒有足夠重視這個mBroadcastRefCount,因此把alarm.operation.send()語句包在了一條if語句中,也就是說在某種狀況下,程序會跳過alarm.operation.send()一句,直接執行下面的語句。然而此時的mBroadcastRefCount還在堅決不移地加一,這直接致使mBroadcastRefCount再也減不到0了,因而mWakeLock也永遠不會釋放了。使人頭痛的是,這個mWakeLock雖然不讓手機深睡眠下去,卻也不會點亮屏幕,因此這個bug潛藏了很久才被找到。還真是應了我說的那句話:「魔鬼總藏在細節中。」
也許一些使用alarmmanager作定時任務的同窗遇到過這樣的問題:設定alarm後,進入設置-->應用程序管理-->強行中止app後,定時任務就失效了。
簡單的講就是:force stop會致使alarm失效。
最典型的例子就是我碰到過的一個bug,使用android手機的時鐘app設置一個鬧鐘,而後進入設置-->應用程序管理裏面,將時鐘這個app force stop掉,結果鬧鐘就不響了。
其實這不是bug,這是android系統的新加入的機制。下面我來詳細分析一下前因後果。
1. 在設置的應用程序管理裏面強行中止app:
這裏會最終會調用到 ActivityManagerService的forceStopPackageLocked()
- private void forceStopPackageLocked(final String packageName, int uid) {
- forceStopPackageLocked(packageName, uid, false, false, true, false);
- Intent intent = new Intent(Intent.ACTION_PACKAGE_RESTARTED,
- Uri.fromParts("package", packageName, null));
- if (!mProcessesReady) {
- intent.addFlags(Intent.FLAG_RECEIVER_REGISTERED_ONLY);
- }
- intent.putExtra(Intent.EXTRA_UID, uid);
- broadcastIntentLocked(null, null, intent,
- null, null, 0, null, null, null,
- false, false, MY_PID, Process.SYSTEM_UID);
- }
代碼裏面發送了一個廣播:ACTION_PACKAGE_RESTARTED,這個廣播大有文章。
最後來看UninstallReceiver,當AlarmManagerService接受到這個廣播後,會把其那些alarm的包名傳過來的給刪除了。
- class UninstallReceiver extends BroadcastReceiver {
- public UninstallReceiver() {
- IntentFilter filter = new IntentFilter();
- filter.addAction(Intent.ACTION_PACKAGE_REMOVED);
- filter.addAction(Intent.ACTION_PACKAGE_RESTARTED);
- filter.addAction(Intent.ACTION_QUERY_PACKAGE_RESTART);
- filter.addDataScheme("package");
- mContext.registerReceiver(this, filter);
- // Register for events related to sdcard installation.
- IntentFilter sdFilter = new IntentFilter();
- sdFilter.addAction(Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE);
- sdFilter.addAction(Intent.ACTION_USER_STOPPED);
- mContext.registerReceiver(this, sdFilter);
- }
-
- @Override
- public void onReceive(Context context, Intent intent) {
- synchronized (mLock) {
- String action = intent.getAction();
- String pkgList[] = null;
- if (Intent.ACTION_QUERY_PACKAGE_RESTART.equals(action)) {
- pkgList = intent.getStringArrayExtra(Intent.EXTRA_PACKAGES);
- for (String packageName : pkgList) {
- if (lookForPackageLocked(packageName)) {
- setResultCode(Activity.RESULT_OK);
- return;
- }
- }
- return;
- } else if (Intent.ACTION_EXTERNAL_APPLICATIONS_UNAVAILABLE.equals(action)) {
- pkgList = intent.getStringArrayExtra(Intent.EXTRA_CHANGED_PACKAGE_LIST);
- } else if (Intent.ACTION_USER_STOPPED.equals(action)) {
- int userHandle = intent.getIntExtra(Intent.EXTRA_USER_HANDLE, -1);
- if (userHandle >= 0) {
- removeUserLocked(userHandle);
- }
- } else {
- if (Intent.ACTION_PACKAGE_REMOVED.equals(action)
- && intent.getBooleanExtra(Intent.EXTRA_REPLACING, false)) {
- // This package is being updated; don't kill its alarms.
- return;
- }
- Uri data = intent.getData();
- if (data != null) {
- String pkg = data.getSchemeSpecificPart();
- if (pkg != null) {
- pkgList = new String[]{pkg};
- }
- }
- }
- if (pkgList != null && (pkgList.length > 0)) {
- for (String pkg : pkgList) {
- //將這個pkg的alarm從AlarmManagerService中去除
- removeLocked(pkg);
- mBroadcastStats.remove(pkg);
- }
- }
- }
- }
- }
爲何google要加入這樣的機制呢?
應該是出於系統安全的考慮,google在4.0系統中在安全方面作了不少努力。
不少病毒程序都不但願本身的進程被用戶強行中止,但願本身的病毒程序能夠一直運行,而常見的方式就是經過設置alarm,在病毒進程被殺死後,經過定時發送廣播來拉起病毒進程,來實現病毒進程的從新啓動。
google也正是看到了這個一點,因此加入了forceStopPackage的這一機制,讓用戶可以有機會幹掉病毒進程。
android系統的安全性一直是android系統的短板,google在提高系統安全性方面也在不斷努力,在以後的文章中,我會再進行介紹。