Android使用记录访问权限详解

使用记录访问权限

什么是使用记录访问权限呢?这是在Android5.0(Api level 21)新添加的,通过该权限我们可以查看设备上其它应用使用情况的统计信息等。

如何使用该权限呢?

首先在manifest中添加:

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

由于该权限默认只授予系统应用,所以添加了ignore属性。

然后通过如下代码进而手动打开权限:

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

当然只要我们在manifest中进行了权限配置,也可以通过设置->安全->有权查看使用情况的应用来打开权限:

到此我们的应用就拥有了该权限。那么有了这个权限到底能做什么呢?继续往下看......

前段时间和同事聊到了一个叫我要当学霸的app,里边有个学习监督的功能,就需要使用记录访问权限,当打开权限后,除了自己和桌面外,其它app都不能正常使用,点击其它app时会直接退到后台并弹出一个提示页面。不妨我们来模拟下这个功能。

在这之前我们首先看一个类UsageStatsManager:

public final class UsageStatsManager {
 public static final int INTERVAL_BEST = 4; //根据提供的开始、结束时间决定时间间隔
 public static final int INTERVAL_DAILY = 0; //以天为时间间隔(最长7天)
 public static final int INTERVAL_MONTHLY = 2; //以月为时间间隔(最长6个月)
 public static final int INTERVAL_WEEKLY = 1; //以周为时间间隔(最长4个星期)
 public static final int INTERVAL_YEARLY = 3; //以年为时间间隔(最长2年)

 UsageStatsManager() {
 throw new RuntimeException("Stub!");
 }

 public List<UsageStats> queryUsageStats(int intervalType, long beginTime, long endTime) {
 throw new RuntimeException("Stub!");
 }

 public List<ConfigurationStats> queryConfigurations(int intervalType, long beginTime, long endTime) {
 throw new RuntimeException("Stub!");
 }

 public UsageEvents queryEvents(long beginTime, long endTime) {
 throw new RuntimeException("Stub!");
 }

 public Map<String, UsageStats> queryAndAggregateUsageStats(long beginTime, long endTime) {
 throw new RuntimeException("Stub!");
 }

 public boolean isAppInactive(String packageName) {
 throw new RuntimeException("Stub!");
 }
}

可以看到该类提供了五种时间间隔类型,这里我们比较关注queryUsageStats()方法,通过该方法我们可以得到一段时间内 其它应用的使用情况。

我们实现思路是这样的,通过UsageStatsManager类获得2秒内手机app的使用数据,找到时间最近的一个,如果不是我们自己的app或桌面则模拟home键点击,同时弹出一个提示页面,具体的代码如下:

private void getTopApp() {
 UsageStatsManager mUsageStatsManager = (UsageStatsManager) getSystemService(Context.USAGE_STATS_SERVICE);//usagestats
 long time = System.currentTimeMillis();
 List<UsageStats> usageStatsList = mUsageStatsManager.queryUsageStats(UsageStatsManager.INTERVAL_BEST, time - 2000, time);

 if (usageStatsList != null && !usageStatsList.isEmpty()) {
  SortedMap<Long, UsageStats> usageStatsMap = new TreeMap<>();
  for (UsageStats usageStats : usageStatsList) {
  usageStatsMap.put(usageStats.getLastTimeUsed(), usageStats);
  }
  if (!usageStatsMap.isEmpty()) {
  String topPackageName = usageStatsMap.get(usageStatsMap.lastKey()).getPackageName();

  if (getLauncherPackageName(mContext).equals(topPackageName) || "com.othershe.test".equals(topPackageName)) {
   return;
  }

  Log.e("TopPackage Name", topPackageName);

  //模拟home键点击
  Intent intent = new Intent(Intent.ACTION_MAIN);
  intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  intent.addCategory(Intent.CATEGORY_HOME);
  startActivity(intent);

  //启动提示页面
  Intent intent1 = new Intent(mContext, TipActivity.class);
  intent1.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
  startActivity(intent1);
  }
 }
 }

因为时间周期是2秒,所以这里我们采用INTERVAL_BEST作为时间间隔。其中的UsageStats对象对应一个查询到的app数据,主要包含以下信息:

getTopApp()是我们的核心方法,当然我们需要开启一个服务,然后在服务中每隔500毫秒执行一次上边的方法,这样就能起到不断检测的作用:

@Override
 public int onStartCommand(Intent intent, int flags, int startId) {

 mTimer = new Timer();
 TimerTask task = new TimerTask() {
  @Override
  public void run() {
  getTopApp();
  }
 };

 mTimer.schedule(task, 1000, 500);
 return super.onStartCommand(intent, flags, startId);
 }

打开权限、启动服务,可以看到实际的运行效果如下,基本符合我们的预期。

类似的道理,我们也可以判断摸个app是否在前台运行。

上边我们使用了INTERVAL_BEST 时间间隔类型,还可以使用其它4中,例如使用INTERVAL_YEARLY:

private void getHistoryApps() {
 Calendar calendar = Calendar.getInstance();
 long endTime = calendar.getTimeInMillis();
 calendar.add(Calendar.YEAR, -1);
 long startTime = calendar.getTimeInMillis();

 UsageStatsManager mUsageStatsManager = (UsageStatsManager) getSystemService(Context.USAGE_STATS_SERVICE);
 List<UsageStats> usageStatsList = mUsageStatsManager.queryUsageStats(UsageStatsManager.INTERVAL_YEARLY, startTime, endTime);

 if (usageStatsList != null && !usageStatsList.isEmpty()) {
  HashSet<String> set = new HashSet<>();
  for (UsageStats usageStats : usageStatsList) {
  set.add(usageStats.getPackageName());
  }

  if (!set.isEmpty()) {
  Log.e("size", set.size() + "");
  }
 }
 }

上边的代码我们最终获得了过去一年手机上使用过的app的包名集合(其中包括系统级别的):

拿到这些包名可以做什么呢?

其实对于网赚类型的应用有这样一种业务场景,就是用户通过下载app来做任务进而赚取收益,但是如果当前设备通过其它网赚应用已经下载过某个app,然后卸载了,再通过你的网赚应用下载。如果你不知道用户之前安装过该app,就需要给用户结算相应的收益,但是你的上游渠道是不会给你结算的,因为这属于同一设备上的重复下载,这样对公司而言就是亏损的。

有了历史包名信息,我们就可以判断用户在一定的时间周期内是否安装过对应的app,进而采取相应的策略,这样可以在一定程度降低损失。当然有个前提,你要友好的引导用户开启权限。

总结

先到这里吧,以上就是这篇文章的全部内容了,更多的用法还有待进一步探究。希望这篇文章对各位Android开发者们能带来一定的帮助,谢谢大家对我们能带来一定的帮助。

demo下载

(0)

相关推荐

  • Android权限控制之自定义权限

    天哪,这篇文章终于说道如何自定义权限了,左盼右盼,其实这个自定义权限相当easy.为了方便叙述,我这边会用到两个app作为例子示范. Permission App: used to define a new permission 这个作为定义权限的App,我称之为Permission App. Client App: used to access the specified activity of Permission App 这个作为访问上述自定义权限的App,我称之为Client App 先

  • 基于android中权限的集合汇总

    程序执行需要读取到安全敏感项必需在androidmanifest.xml中声明相关权限请求, 完整列表如下: 1. android.permission.ACCESS_CHECKIN_PROPERTIES    允许读写访问"properties"表在 checkin数据库中,改值可以修改上传( Allows read/write access to the "properties" table in the checkin database, to change

  • Android需要提升权限的操作方法

    权限提升方法:一种方法:1.在AndroidManifest.xml中的manifest节点中添加 android:sharedUserId="android.uid.system".2.添加后程序在虚拟机上是不可以直接用的.但可以用eclipse编译成apk.3.(这一步经验证不执行即可)编译成apk后用压缩工具打开apk,把META-INF目录中的CERT.SF.CERT.RSA 两个文件删除.4.使用android自带的签名工具signapk.jar 以及源码中的platform

  • Android 访问文件权限的四种模式介绍

    Linux文件的访问权限 * 在Android中,每一个应用是一个独立的用户 * drwxrwxrwx * 第1位:d表示文件夹,-表示文件 * 第2-4位:rwx,表示这个文件的拥有者(创建这个文件的应用)用户对该文件的权限 * r:读 * w:写 * x:执行 * 第5-7位:rwx,表示跟文件拥有者用户同组的用户对该文件的权限 * 第8-10位:rwx,表示其他用户组的用户对该文件的权限 openFileOutput的四种模式 * MODE_PRIVATE:-rw-rw---- * MOD

  • Android 操作系统获取Root权限 原理详细解析

    android root权限破解分析 许多机友新购来的Android机器没有破解过Root权限,无法使用一些需要高权限的软件,以及进行一些高权限的操作,其实破解手机Root权限是比较简单及安全的,破解Root权限的原理就是在手机的/system/bin/或/system/xbin/目录下放置一个可执行文件"su",这是一个二进制文件,相当于电脑上的exe文件,仅仅在系统中置入这个"su"文件是不会给手机的软件或硬件造成任何故障. 下面的代码是android系统原版的

  • android 权限大全 分享

    访问登记属性 android.permission.ACCESS_CHECKIN_PROPERTIES ,读取或写入登记check-in数据库属性表的权限 获取错略位置 android.permission.ACCESS_COARSE_LOCATION,通过WiFi或移动基站的方式获取用户错略的经纬度信息,定位精度大概误差在30~1500米获取精确位置 android.permission.ACCESS_FINE_LOCATION,通过GPS芯片接收卫星的定位信息,定位精度达10米以内访问定位额

  • Android 6.0权限申请详解及权限资料整理

    在android 6.0开始,部分的权限需要我们动态申请,也就是说当我们的打开app的时候系统不会主动像您申请app所需要的部分权限,需要客户在使用app的时候主动的去申请. 一.权限的申请两步骤: 1.权限申请: /** * @param permissions需要申请的权限 * @param requestCode申请回调code */ public static void requestPermissions(final @NonNull Activity activity,final @

  • Android获取ROOT权限的实例代码

    获取Android的ROOT权限其实很简单,只要在Runtime下执行命令"su"就可以了. 复制代码 代码如下: // 获取ROOT权限public void get_root(){ if (is_root()){        Toast.makeText(mCtx, "已经具有ROOT权限!", Toast.LENGTH_LONG).show();    }    else{        try{            progress_dialog = P

  • Android使用记录访问权限详解

    使用记录访问权限 什么是使用记录访问权限呢?这是在Android5.0(Api level 21)新添加的,通过该权限我们可以查看设备上其它应用使用情况的统计信息等. 如何使用该权限呢? 首先在manifest中添加: <uses-permission android:name="android.permission.PACKAGE_USAGE_STATS" tools:ignore="ProtectedPermissions" /> 由于该权限默认只授

  • Java中成员方法与成员变量访问权限详解

    记得在一次面试的笔试题中,有的面试官会要求写出具体的像pullic这些访问限定符的作用域.其实,平常我都没去系统的考虑这些访问限定符的作用域,特别是包内包外的情况,OK,笔试不行了. 这是java基本的知识,也是公司看重的,那没办法啦,我的脑袋记不住东西,那我只能把这些东西写下来方便自己温故知新,不废话了,贴代码了. 代码如下: package com.jaovo; /** *_1_ 成员变量访问权限的求证 * public private protected default(默认的权限) *自

  • ASP.NET Core使用自定义验证属性控制访问权限详解

    前言 大家都知道在应用中,有时我们需要对访问的客户端进行有效性验证,只有提供有效凭证(AccessToken)的终端应用能访问我们的受控站点(如WebAPI站点),此时我们可以通过验证属性的方法来解决. 本文将详细介绍ASP.NET Core使用自定义验证属性控制访问权限的相关内容,分享出来供大家参考学习,下面话不多说了,来一起看看详细的介绍吧 方法如下 一.public class Startup的配置: //启用跨域访问(不同端口也是跨域) services.AddCors(options

  • Android Binder 通信原理图文详解

    目录 前言 1. Binder的作用 2. 进程与Binder驱动如何通信 3. ServiceManager进程的作用 Binder Client.Binder Server.ServiceManager关系 ServiceManager注册进Binder 4. 进程添加服务到ServiceManager的流程 其它进程找到SM 添加服务到ServiceManager BBinder作用 5. 进程从ServiceManager获取服务的流程 其它进程找到SM 从ServiceManager获

  • Android pdf viewer在android studio应用问题说明详解

    之前一直是做.NET开发的,最近需要弄一个新闻app,能力有限,只能借助HTML5 WebAPP+android studio来完成这项工作. android studio主要用WebView来加载发布好的WebApp,打包生产APP. 其中由于显示一些pdf文档,所以研究了一下,记录一下心得,同时也希望帮助到新手们. android 显示网络pdf,基本原理:先将pdf文件通过DownloadManager下载到手机sdk某个文件夹中,然后通过android-pdf-viewer插件进行显示.

  • Android 中ContentProvider的实例详解

    Android 中ContentProvider的实例详解 Content Provider 的简单介绍: * Android中的Content Provider 机制可支持在多个应用中存储和读取数据.这也是跨应用 共享数据的唯一方式.在Android系统中,没有一个公共的内存区域,供多个应用共享存储数据: * Android 提供了一些主要数据类型的ContentProvider ,比如:音频.视频.图片和私人通讯录等: 在android.provider 包下面找到一些android提供的C

  • Android 中Manifest.xml文件详解

    Android 中Manifest.xml文件详解 每一个Android项目都包含一个清单(Manifest)文件--AndroidManifest.xml,它存储在项目层次中的最底层.清单可以定义应用程序及其组件的结构和元数据. 它包含了组成应用程序的每一个组件(活动.服务.内容提供器和广播接收器)的节点,并使用Intent过滤器和权限来确定这些组件之间以及这些组件和其他应用程序是如何交互的. 它还提供了各种属性来详细地说明应用程序的元数据(如它的图标或者主题)以及额外的可用来进行安全设置和单

  • Android 判断网络状态实例详解

    Android 判断网络状态实例详解 实例代码 package com.example.android; import java.io.IOException; import java.net.HttpURLConnection; import java.net.InetAddress; import java.net.NetworkInterface; import java.net.SocketException; import java.net.URL; import java.util.

  • Android组件之服务的详解

    目录 一.服务的概念 二.Android的多线程编程 2.1 线程的基本用法 2.2 在子线程中更新UI 更新方式一 更新方式二 2.3 解析异步消息处理机制 2.4 使用AsyncTask 三.服务的基本用法 3.1 首先定义一个服务 3.2 MyService类里重写几个方法 3.3 在注册文件中完成对服务的注册 3.4 启动和停止服务 3.5 活动和服务进行通信 四.服务的生命周期 五.服务的更多技巧 5.1 使用前台服务 5.2 服务中的多线程问题&IntentService 一.服务的

  • Linux 下sudo网络权限详解

    Linux 下sudo网络权限详解 对于设置了网络代理的服务器,在当前用户下执行网络访问没有问题,但通过sudo执行命令时,就会出现"无网络连接"的错误. 背景 对于设置了网络代理的服务器,在当前用户下执行网络访问没有问题,但通过sudo执行命令时,就会出现"无网络连接"的错误. 普通权限下,wget成功. # wget https://github.com --2016-12-08 09:00:43-- https://github.com/ Connecting

随机推荐