天天看點

判斷App是否在背景運作

在一些場景中,經常會需要判斷App是否在背景運作,比如是否顯示解鎖界面,收到新消息是否顯示Notification等。需求可能是多樣化的,但所依仗的原理是相通的,今天Stay打算說說這些需求的最優解。

當然,Stay肯定不會說去for loop判斷目前runningProcess或者runningTasks。比如:

這樣

判斷App是否在背景運作

或者這樣

判斷App是否在背景運作

這種方法調用起來感覺就像是在用Windows系統裡的任務管理器,真是讓人蛋疼。我們暫且不去計較性能問題,就說為啥Android連個像樣的API都不給我,着實讓人郁悶。

如果帶着這樣的質疑去調研,你會發現還真有其他方式來實作。

Android在SDK 14的時候提供了一個Callback。ActivityLifecycleCallbacks,你可以通過這個Callback拿到App所有Activity的生命周期回調。看圖:

這個Callback寫在Application裡的,你可以在Application初始化的時候來注冊。我們可以寫個單例類來cache這些status。這裡我叫它AppStatusTracker。在Application的onCreate()裡讓AppStatusTracker注冊ActivityLifecycleCallbacks。

判斷App是否在背景運作

拿到這些Callback有什麼用呢,我怎麼能知道App是否在前台運作呢。 

别急,我們先來說說Activity的生命周期。這是面試時必問題,雖然有官方答案,但真正了解生命周期,并靈活運用的不多。

我們來設想下如果Activity調用了onResume(),那麼這個Activity肯定是可見的,也就是運作在前台的。如果調用了onPause(),且沒有Activity來調用onResume(),那麼App要跑到背景去了。至于它是點了home鍵還是back鍵我都不管。

通過這樣的判斷,我們來利用ActivityLifecycleCallbacks回調的onActivityResumed()和onActivityPaused()方法來計數,如果隻有一個activityCount,那麼目前App在前台,如果木有activityCount,它就在背景。

好了,就這麼愉快的解決了,再也不用for loop了。但是很快你會發現,這裡有個延時,會導緻判斷不準确。

我們假設有兩個Activities,一個A,一個B,從A跳轉到B,生命周期怎麼走的? A.onPause() -> B.onResume() 對應到ActivityLifecycleCallbacks裡是onActivityPaused(A) -> onActivityResumed(B),剛才我們說的計數resumeCount,在onActivityPaused()裡--,在onActivityResumed()裡++, 根據這樣的判斷會有個短暫的間隔,也就是在A的onPause()到B的onResume()之間,App是運作在背景的,這樣邏輯肯定就不對了。

那如何解決問題呢?如果你列印過生命周期的哪些方法,你會發現是Activity間切換的步驟是這樣的: 

判斷App是否在背景運作

從WelcomeActivity跳轉到GestureActivity。(這裡隻說onStart, onResume這些回調 )

A.onPause() -> B.onStart() -> B.onResume() -> A.onStop()

我估摸着60%的同學都沒想過Activities之間切換的生命周期是什麼樣的。

通過這些回調我們可以将這個計數放在onStart()和onStop()中去,這樣就不會存在那個短暫間隔。activityCount==1,那麼就是前台,activityCount==0,那就是背景。這樣判斷很很簡單了吧。

現在再說,什麼情況下來顯示手勢解鎖界面。 

我的需求是當使用者鎖屏後再解鎖或者切換到背景10分鐘後顯示手勢解鎖界面。 

我們拆分下需求,先說鎖屏,解鎖。

這個是有BroadCastReciever來接收的,注冊下就可以了,每次收到鎖屏ACTION_SCREEN_OFF的action時,将AppStatusTracker裡的isScreenOff設定為true。 

當onActivityResumed()被調用時再将isScreenOff設為false。

再說切換到背景10分鐘後顯示手勢解鎖。這個隻需要在onActivityStop()時更新下lastBackgroudTimestamp就可以了