应用程序在前台时如何计算应用程序使用时间?
How to count app usage time while app is on foreground?
我正在开发一个 android 应用程序来跟踪应用程序的日常使用情况。这个想法是用户可以为任何应用程序设置每日时间限制,并且在超过限制后最多 2 分钟内会出现通知。 (延迟的原因:我已经使用 AlarmManager class 创建了一个警报系统,它会每分钟响一次 运行 a JobIntentService 这将检查是否超过了任何应用程序的限制)
我使用 queryEvents method of UsageStatsManager class 来计算应用程序的使用时间。
这是我计算应用程序使用时间的代码(我已经关注 ):
HashMap<String, Integer> getTimeSpent(Context context, String packageName, long beginTime, long endTime) {
UsageEvents.Event currentEvent;
List<UsageEvents.Event> allEvents = new ArrayList<>();
HashMap<String, Integer> appUsageMap = new HashMap<>();
UsageStatsManager usageStatsManager = (UsageStatsManager)context.getSystemService(Context.USAGE_STATS_SERVICE);
UsageEvents usageEvents = usageStatsManager.queryEvents(beginTime, endTime);
while (usageEvents.hasNextEvent()) {
currentEvent = new UsageEvents.Event();
usageEvents.getNextEvent(currentEvent);
if(currentEvent.getPackageName().equals(packageName) || packageName == null) {
if (currentEvent.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED
|| currentEvent.getEventType() == UsageEvents.Event.ACTIVITY_PAUSED) {
allEvents.add(currentEvent);
String key = currentEvent.getPackageName();
if (appUsageMap.get(key) == null)
appUsageMap.put(key, 0);
}
}
}
for (int i = 0; i < allEvents.size() - 1; i++) {
UsageEvents.Event E0 = allEvents.get(i);
UsageEvents.Event E1 = allEvents.get(i + 1);
if (E0.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED
&& E1.getEventType() == UsageEvents.Event.ACTIVITY_PAUSED
&& E0.getClassName().equals(E1.getClassName())) {
int diff = (int)(E1.getTimeStamp() - E0.getTimeStamp());
diff /= 1000;
Integer prev = appUsageMap.get(E0.getPackageName());
if(prev == null) prev = 0;
appUsageMap.put(E0.getPackageName(), prev + diff);
}
}
return appUsageMap;
}
简而言之,上面的代码计算了应用程序进入前台时的时间戳(UsageEvents.Event.ACTIVITY_RESUMED)和进入后台时的时间戳(UsageEvents.Event.ACTIVITY_PAUSED)。然后它将这个差异添加到应用程序的总使用时间。
问题是除非应用程序进入后台,否则无法计算在前台花费的时间。因此,如果超过使用限制,直到应用进入后台后才会显示通知。
是否真的可以在应用程序处于前台时获取前台时间?
N.B。我已经尝试 queryUsageStats along with UsageStats.getTotalTimeInForeground() 但未能成功,因为 queryUsageStats 有一些与此问题无关的其他问题。
我已经解决了这个问题。
添加当前 运行 应用进入前台的当前时间和时间戳的差异就可以了。
我刚刚在 return 语句之前添加了以下代码:
UsageEvents.Event lastEvent = allEvents.get(allEvents.size() - 1);
if(lastEvent.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED) {
int diff = (int)System.currentTimeMillis() - (int)lastEvent.getTimeStamp();
diff /= 1000;
Integer prev = appUsageMap.get(lastEvent.getPackageName());
if(prev == null) prev = 0;
appUsageMap.put(lastEvent.getPackageName(), prev + diff);
}
很简单,我应该在发问题之前考虑一下。
我正在开发一个 android 应用程序来跟踪应用程序的日常使用情况。这个想法是用户可以为任何应用程序设置每日时间限制,并且在超过限制后最多 2 分钟内会出现通知。 (延迟的原因:我已经使用 AlarmManager class 创建了一个警报系统,它会每分钟响一次 运行 a JobIntentService 这将检查是否超过了任何应用程序的限制)
我使用 queryEvents method of UsageStatsManager class 来计算应用程序的使用时间。
这是我计算应用程序使用时间的代码(我已经关注
HashMap<String, Integer> getTimeSpent(Context context, String packageName, long beginTime, long endTime) {
UsageEvents.Event currentEvent;
List<UsageEvents.Event> allEvents = new ArrayList<>();
HashMap<String, Integer> appUsageMap = new HashMap<>();
UsageStatsManager usageStatsManager = (UsageStatsManager)context.getSystemService(Context.USAGE_STATS_SERVICE);
UsageEvents usageEvents = usageStatsManager.queryEvents(beginTime, endTime);
while (usageEvents.hasNextEvent()) {
currentEvent = new UsageEvents.Event();
usageEvents.getNextEvent(currentEvent);
if(currentEvent.getPackageName().equals(packageName) || packageName == null) {
if (currentEvent.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED
|| currentEvent.getEventType() == UsageEvents.Event.ACTIVITY_PAUSED) {
allEvents.add(currentEvent);
String key = currentEvent.getPackageName();
if (appUsageMap.get(key) == null)
appUsageMap.put(key, 0);
}
}
}
for (int i = 0; i < allEvents.size() - 1; i++) {
UsageEvents.Event E0 = allEvents.get(i);
UsageEvents.Event E1 = allEvents.get(i + 1);
if (E0.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED
&& E1.getEventType() == UsageEvents.Event.ACTIVITY_PAUSED
&& E0.getClassName().equals(E1.getClassName())) {
int diff = (int)(E1.getTimeStamp() - E0.getTimeStamp());
diff /= 1000;
Integer prev = appUsageMap.get(E0.getPackageName());
if(prev == null) prev = 0;
appUsageMap.put(E0.getPackageName(), prev + diff);
}
}
return appUsageMap;
}
简而言之,上面的代码计算了应用程序进入前台时的时间戳(UsageEvents.Event.ACTIVITY_RESUMED)和进入后台时的时间戳(UsageEvents.Event.ACTIVITY_PAUSED)。然后它将这个差异添加到应用程序的总使用时间。
问题是除非应用程序进入后台,否则无法计算在前台花费的时间。因此,如果超过使用限制,直到应用进入后台后才会显示通知。
是否真的可以在应用程序处于前台时获取前台时间?
N.B。我已经尝试 queryUsageStats along with UsageStats.getTotalTimeInForeground() 但未能成功,因为 queryUsageStats 有一些与此问题无关的其他问题。
我已经解决了这个问题。
添加当前 运行 应用进入前台的当前时间和时间戳的差异就可以了。
我刚刚在 return 语句之前添加了以下代码:
UsageEvents.Event lastEvent = allEvents.get(allEvents.size() - 1);
if(lastEvent.getEventType() == UsageEvents.Event.ACTIVITY_RESUMED) {
int diff = (int)System.currentTimeMillis() - (int)lastEvent.getTimeStamp();
diff /= 1000;
Integer prev = appUsageMap.get(lastEvent.getPackageName());
if(prev == null) prev = 0;
appUsageMap.put(lastEvent.getPackageName(), prev + diff);
}
很简单,我应该在发问题之前考虑一下。