本文通過圖文並茂的方式給大家介紹android判斷app狀態的相關內容,具體詳情如下所示:
要瞭解這塊,首先需要明白一些概念,app,process,task
1.process就是進程,是linux的概念。
2.一般一個app擁有一個uid,運行在一個進程裡,如果app中給service等定義不同的uid,那Service就運行在另外一個進程裡,也就是說uid就相當於進程的id一樣,一個uid就代表一個進程;也可以幾個app定義一個uid,那他們就運行在一個進程裡了。
3.task是android系統的一個activity的棧,包含多個app的activity,通過ActivityManager可以擷取棧中的activity資訊,從而判斷activity對應應用的狀態。
3.1可以做的事情包括:
ActivityManagermActivityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);PackageManagermPackageManager = context.getPackageManager();
①getRecentTasks() 擷取最近開啟的task,手機查看最近開啟的應用可以用這個實現。
②getRunningAppProcess() 擷取app運行中的process。
List list = mActivityManager.getRunningAppProcesses();StringBuilder apps = new StringBuilder();for (RunningAppProcessInfo info : list) {apps.append(info.processName + "\n" + info.importance);}System.out.println(apps.toString());
list是系統當前運行進程的集合,importance屬性==100 表示這個進程在前台,其他數字表示在後台,所以通過importance和processName判斷應用是否在前台
我做測試的時候,方法返回的list中只有這個應用的進程,前台是狀態是100,後台時狀態是400,
這是因為My Phone是5.0以上的,從api21開始這個方法只能返回自身應用的進程資訊,如果應用中有Service並且定義了process屬性,那麼就會返回兩個進程,app進程仍是100,Service是其他值。5.0以下可以返回所有,另外有個方法5.0以上可以返回所有,5.0以下不行,後面說。
*另外:當某個app的Service設定為粘性時,Service所在的app就會被認為是在前台,app進程的importance一直是100,這個時候就不能通過這個方法判斷是否在前台了。
總結:5.0以上只能判斷自身應用是否在前台,當有app的Service被設定為粘性時,就不能用了 。
If(importance == 100) 中100在runningprocessinfo下的常量IMPORTANCE_FOREGROUND
所以這裡要寫成 if(info.importance == RunningProcessInfo.IMPORTANCE_FOREGROUND)
③getRunningServices()擷取系統運行中的後台service。
④getRunningTask()擷取系統運行中的任務。
上面說了activity的task棧,棧頂activity所在app就是前台的app,所有拿到棧頂activity的資訊擷取他的包名,對比應用的包名來判斷應用是不是在前台,5.0以前可以判斷是哪個應用在前台,5.0以後只會返回自身和部分不重要task,不能再判斷其他應用是否在前台,只能判斷自己了。
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);List tasks = am.getRunningTasks(1);if (!tasks.isEmpty()) {ComponentName topActivity = tasks.get(0).topActivity;if (topActivity.getPackageName().equals(context.getPackageName())) {“前台”}}
“後台”
4.ActivityLifecycleCallbacks
以前做過所有的activity都整合baseactivity,重寫onCreate方法,onStop方法,定義一個變數count統計開啟的activity數,onCreate時 count++,onStop時count--,當count == 0 時,應用就在後台了。現在,從api14開始,api提供了一個所有activity生命週期的回調,就是ActivityLifecycleCallbacks,每個activity的生命週期都會觸發,通過registerActivityLifecycleCallbacks註冊這個回調。
其實和重寫baseactivity的onCreate和onStop一樣的,在回調的onCreate中count++,onStop中count--,在onStop中增加個判斷,if(count == 0){“我在後台了”},為什麼在onStop中呢?不論是back返回還是home退出,還是其他手段退出,activity總會回調onStop的。
也可以在onStop中直接調用getRunningAppProcess或者getRunningTask判斷應用是不是跑後台去了,這個時候可以做一些事情。然後從後台回到前台調用onCreate,又可以做一些事情。
我做過的這個是在每次應用跑到前台時就檢測是不是有新版本更新。在自訂application的onCreate中註冊這個回調。
5.UsageStatsManager ,這個就是上面說的5.0以後才有效方法.使用需要加入許可權
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions" xmlns:tools="http://schemas.android.com/tools"></uses-permission>
手機-設定-安全-進階,在有權查看使用方式的應用中,選中自己應用。
6.通過Android內建的無障礙功能,
http://effmx.com/articles/tong-guo-android-fu-zhu-gong-neng-accessibility-service-jian-ce-ren-yi-qian-tai-jie-mian/
7.讀取Linux核心保護在/proc目錄下的process進程資訊,GitHub地址https://github.com/jaredrummler/AndroidProcesses
不需要許可權,可以判斷任意應用是否在前台,但是檢索檔案夾需要時間。
用法
擷取一系列正在啟動並執行App的進程
List processes = ProcessManager.getRunningAppProcesses();</androidappprocess>
擷取任一正在啟動並執行App進程的詳細資料
AndroidAppProcess process = processes.get(location);String processName = process.name;Stat stat = process.stat();int pid = stat.getPid();int parentProcessId = stat.ppid();long startTime = stat.stime();int policy = stat.policy();char state = stat.state();Statm statm = process.statm();long totalSizeOfProcess = statm.getSize();long residentSetSize = statm.getResidentSetSize();PackageInfo packageInfo = process.getPackageInfo(context, 0);String appName = packageInfo.applicationInfo.loadLabel(pm).toString();
判斷是否在前台
if (ProcessManager.isMyProcessInTheForeground()) {// do stuff}
擷取一系列正在啟動並執行App進程的詳細資料
List processes = ProcessManager.getRunningAppProcessInfo(ctx);</activitymanager.runningappprocessinfo>
以上內容是針對android判斷app 狀態,希望對大家有所協助!