判斷App是否在后台運行


 

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

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

這樣

或者這樣

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

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

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

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

拿到這些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間切換的步驟是這樣的: 


從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就可以了

核心代碼如下:

原理很簡單,但是涉及到的知識點很重要,大家可以自己寫寫測試下,別總依賴別人的代碼,別人的類庫,技術實現很簡單,但需求的變體擴展有時候還是需要自己來想辦法解決的。

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM