Android6.0以后,如何获取当前运行的app

前言

在Android5.0,即Lollipop(api level 21)之前,大家都幸福的使用如下代码来获得当前运行的app,即所谓的top Activity:

ActivityManager activityManager = (ActivityManager) this.getSystemService(Context.ACTIVITY_SERVICE);
ComponentName cn = activityManager.getRunningTasks(1).get(0).topActivity;

自从Lollipo以泄露用户信息的理由彻底禁止了getRunningTasks方法之后,世道就变得艰难起来,StackOverFlow上曾经大张旗鼓的讨论此问题,很多人使用ActivityManager.RunningAppProcessInfo方法来获取顶层app,这个方法似乎在某个版本中是有效的。但是幸福总是短暂的,到了Android6.0版本,即Marshmallow(api level 23)时,这些方法统统的废了,除了自己app中的信息外,只能获取启动器的信息。道理是很简单的,Marshmallow以权限严格著称,因此对于这种可能泄露其他应用信息的方法一概禁止了。
取而代之的是下一章要说的方法。

UsageStatsManager

UsageStatsManager是用来统计app使用情况的类,直到Lollipop(api level 21)才加入Android。此类的具体用法可以参考:
http://developer.android.com/reference/android/app/usage/UsageStatsManager.html
本文仅介绍如何使用它来获取当前运行的app。

第一步,修改AndroidManifest.xml,添加权限

如下标红的部分是要添加的权限:

<?xml version="1.0" encoding="utf-8"?> <!--注意这里:添加xmlns:tools="http://schemas.android.com/tools"--> <manifest xmlns:android="http://schemas.android.com/apk/res/android" xmlns:tools="http://schemas.android.com/tools" package="com.dumaisoft.wxb.gettopactivity"> <application  ... </application> <!--注意这里:添加权限--> <uses-permission  android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions"/> </manifest>

第二步,检测并引导用户开启权限

从Marshmallow开始,用户可以一个一个的给app授权,意味着系统对权限的管理更加严格了。这里,我们的app必须让用户开启“Apps with usage access”权限。
检测用户是否开启权限的代码如下:

    //检测用户是否对本app开启了“Apps with usage access”权限
    private boolean hasPermission() {
        AppOpsManager appOps = (AppOpsManager)
                getSystemService(Context.APP_OPS_SERVICE);
        int mode = 0;
        if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT) {
            mode = appOps.checkOpNoThrow(AppOpsManager.OPSTR_GET_USAGE_STATS,
                    android.os.Process.myUid(), getPackageName());
        }
        return mode == AppOpsManager.MODE_ALLOWED;
    }

引导用户开启权限的代码如下:

    private static final int MY_PERMISSIONS_REQUEST_PACKAGE_USAGE_STATS = 1101;

    @Override
    protected void onActivityResult(int requestCode, int resultCode, Intent data) {
        if (requestCode == MY_PERMISSIONS_REQUEST_PACKAGE_USAGE_STATS) {
            if (!hasPermission()) {
                //若用户未开启权限,则引导用户开启“Apps with usage access”权限
                startActivityForResult(
                        new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS),
                        MY_PERMISSIONS_REQUEST_PACKAGE_USAGE_STATS);
            }
        }
    }

第三步,使用UsageStatsManager来获取当前运行的app

为了便于看到结果,我们将代码写在一个Service中,并利用Logcat输出结果,这样可以切换运行的app,并实时看到结果,主要的代码如下:

 private void getTopApp(Context context) {
            if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
                UsageStatsManager m = (UsageStatsManager) context.getSystemService(Context.USAGE_STATS_SERVICE);
                if (m != null) {
                    long now = System.currentTimeMillis();
                    //获取60秒之内的应用数据
                    List<UsageStats> stats = m.queryUsageStats(UsageStatsManager.INTERVAL_BEST, now - 60 * 1000, now);
                    Log.i(TAG, "Running app number in last 60 seconds : " + stats.size());

                    String topActivity = "";

                    //取得最近运行的一个app,即当前运行的app
                    if ((stats != null) && (!stats.isEmpty())) {
                        int j = 0;
                        for (int i = 0; i < stats.size(); i++) {
                            if (stats.get(i).getLastTimeUsed() > stats.get(j).getLastTimeUsed()) {
                                j = i;
                            }
                        }
                        topActivity = stats.get(j).getPackageName();
                    }
                    Log.i(TAG, "top running app is : "+topActivity);
                }
            }
        }

输出的结果

下面是使用安装了Marshmallow的模拟器输出的结果,传说中一些手机无法支持UsageStatsManager,我并没有一一检测:


04-24 13:28:06.041 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.example.android.apis
04-24 13:28:08.082 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:08.082 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.example.android.apis
04-24 13:28:10.111 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:10.111 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.google.android.googlequicksearchbox
04-24 13:28:12.117 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:12.117 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.google.android.googlequicksearchbox
04-24 13:28:14.155 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:14.155 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.android.dialer
04-24 13:28:16.190 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:16.190 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.android.dialer
04-24 13:28:18.232 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:18.232 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.google.android.googlequicksearchbox
04-24 13:28:20.279 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:20.279 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.android.messaging
04-24 13:28:22.311 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: Running app number in last 60 seconds : 24
04-24 13:28:22.311 25119-25239/com.dumaisoft.wxb.gettopactivity I/TestService: top running app is : com.google.android.googlequicksearchbox

获取源代码

完整的源代码放到github上了,项目地址为:
https://github.com/apkkids/GetTopActivity
我的开发环境是Android Studio2.0,如果开发环境不同可能会导致本代码无法在你的机器上运行,这时你只需要看其中的源码就好。
另:本文中大量参考了StackOverFlow中的代码,出处就不一一列出了,Sorry。希望能够帮到大家。

你可能感兴趣的:(android,topActivit,UsageStats)