[Android] Android5.1系统自带的应用启动次数统计

reference to : http://blog.csdn.net/elder_sword/article/details/50508257

前段时间要做一个统计手机中激活量的东东,这个统计不是单独的某个应用统计,是整个手机中所有预装应用的安装量。第一时间想到的就是后台去跑个 service,隔一段时间去扫一遍,和程序锁原理类似。但是这种方案有个弊端,大家也都了解,就是费电,当然也有优点,那就是统计精确。还有另外一种方 案,就是android自带的统计,之前也只是粗略的知道有这么回事,没有仔细看过。

具体操作,在拨号界面输入 ×#×#4636#×#×,手机会自动跳转到测试页面,然后点击使用情况统计数据,你就会看到统计的界面了。一开始感觉这个需求应该好做了,但仔细一看不 对啊,5.1的手机上面就没有打开次数这一栏了,5.0之前是好用的,没关系这里打开次数下面会说到的。说了一堆,下面上代码。

首先是UsageStatsManager这个类,先看下介绍
这里写图片描述

这里说的很清楚 需要android.permission.PACKAGE_USAGE_STATS这个权限,而且是system级别,不能在第三方应用中使用,但是在Settings里面有个选项是可以赋给你的应用查询的权限。
这里写图片描述

但前提你必须把这个权限写到你的manifest里面,否则在这个列表回看不到你的应用。这时你会发现manifest报错,导致应用不能直接运行,没关系,看代码。

<uses-permission  android:name="android.permission.PACKAGE_USAGE_STATS"  tools:ignore="ProtectedPermissions"/>

这样写可以让你代码正常运行

获取统计列表

public static List<UsageStats> getUsageStatsList(Context context){
        UsageStatsManager usm = getUsageStatsManager(context);
        Calendar calendar = Calendar.getInstance();
        long endTime = calendar.getTimeInMillis();
        calendar.add(Calendar.YEAR, -1);
        long startTime = calendar.getTimeInMillis();
        Log.d(TAG, "Range start:" + dateFormat.format(startTime) );
        Log.d(TAG, "Range end:" + dateFormat.format(endTime));
        List<UsageStats> usageStatsList = usm.queryUsageStats(UsageStatsManager.INTERVAL_DAILY,startTime,endTime);
        return usageStatsList;
}
@SuppressWarnings("ResourceType")
    private static UsageStatsManager getUsageStatsManager(Context context) {
        UsageStatsManager usm = (UsageStatsManager) context.getSystemService("usagestats");
        return usm;
    }

这个方法第一次是空的,因为你没有权限,判断为空之后去跳转到Settings,这里大家看到了,我传的startTime,是过去一年的,而这个方法统计是有时间段的,上面的注释也有说。我统计的是过去一年到当前的所有应用的使用次数。

Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS);
startActivity(intent);

赋完权限之后就可以正常获取到列表
打印所需要的值

public static void printUsageStats(List<UsageStats> usageStatsList) throws IllegalAccessException, IllegalArgumentException, NoSuchFieldException{
        for (UsageStats u : usageStatsList){
            Log.d(TAG, "Pkg: " + u.getPackageName() + "\t     " + "ForegroundTime: "
                    + DateUtils.formatElapsedTime(u.getTotalTimeInForeground() / 1000)+"   lasttimeuser:"+DateUtils.formatSameDayTime(u.getLastTimeUsed(),
                            System.currentTimeMillis(), DateFormat.MEDIUM, DateFormat.MEDIUM)
                    +"times"+u.getClass().getDeclaredField("mLaunchCount").getInt(u));
}

这里大家看见了,我获取打开次数mLaunchCount的方法,没错,就是反射。到这里,一般第三方应用的需求应该可以满足了,但是系统类的应用要去手动打开权限未免有些突兀。我又跟代码到对应switch打开的代码,结合自己应用的需求

AppOpsManager mAppOpsManager = (AppOpsManager) context.getSystemService(Context.APP_OPS_SERVICE);
PackageInfo packageInfo = context.getPackageManager().getPackageInfo(context.getPackageName(),PackageManager.GET_PERMISSIONS);
mAppOpsManager.setMode(43, packageInfo.applicationInfo.uid,context.getPackageName(), AppOpsManager.MODE_ALLOWED);

上面这段代码可以让系统应用自动有权限。有一段setMode(43, 这里的43是抄之前switch打开的代码

致此,所有代码结束。最后顺带说一句,我这里是用来统计激活量,所以打开次数要求不是很精确,这种方式可以满足,但是要求统计非常精确的需要再考虑 一下。因为这种方式拿到的打开次数应该是和自身的其他service相关,比如已很很简单的demo,它的打开次数是精确的,但是如果带有一个 service,打开次数就会以2增长,第一次为2,第二次为4,这里我是自己理解的,代码跟到后面能力有限,没有找到真正的原因,还请大家谅解。

posted @ 2016-02-19 22:25  demoblog  阅读(9247)  评论(0编辑  收藏  举报