在Android L中,Google已禁用getRunningTasks。现在它只能返回自己的应用程序任务和主启动器。我再也无法获得其他应用程序任务了。 我们的应用需要该方法来确定当前的顶级应用。 有人有另一种方法吗?
我在谷歌搜索过,除此之外没有其他主题: https://code.google.com/p/android-developer-preview/issues/detail?id=29
答案 0 :(得分:45)
对于我最近开展的项目,我还需要检测某些应用程序何时启动。我的所有研究都引出了getRunningTasks方法,该方法从Lollipop开始不推荐使用。 然而,令我惊讶的是,我发现一些应用程序锁定应用仍然适用于棒棒糖设备,因此他们必须提出解决方案来解决这个问题。所以我挖了一点。这是我发现的:
用于检测默认日历应用启动时间的一些示例代码:
public class DetectCalendarLaunchRunnable implements Runnable {
@Override
public void run() {
String[] activePackages;
if (Build.VERSION.SDK_INT > Build.VERSION_CODES.KITKAT_WATCH) {
activePackages = getActivePackages();
} else {
activePackages = getActivePackagesCompat();
}
if (activePackages != null) {
for (String activePackage : activePackages) {
if (activePackage.equals("com.google.android.calendar")) {
//Calendar app is launched, do something
}
}
}
mHandler.postDelayed(this, 1000);
}
String[] getActivePackagesCompat() {
final List<ActivityManager.RunningTaskInfo> taskInfo = mActivityManager.getRunningTasks(1);
final ComponentName componentName = taskInfo.get(0).topActivity;
final String[] activePackages = new String[1];
activePackages[0] = componentName.getPackageName();
return activePackages;
}
String[] getActivePackages() {
final Set<String> activePackages = new HashSet<String>();
final List<ActivityManager.RunningAppProcessInfo> processInfos = mActivityManager.getRunningAppProcesses();
for (ActivityManager.RunningAppProcessInfo processInfo : processInfos) {
if (processInfo.importance == ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND) {
activePackages.addAll(Arrays.asList(processInfo.pkgList));
}
}
return activePackages.toArray(new String[activePackages.size()]);
}
}
注意:getRunningAppProcesses也用于调试或构建面向用户的流程管理UI&#34;。不确定google是否会以类似的方式关闭此后门,以获得getRunningTasks。
所以不,你不能再获得topActivity了。但是有点破解你可以获得类似的结果。
答案 1 :(得分:33)
正如MKY所提到的,getRunningTasks()
方法无法在Lollipop中获取当前应用程序。
正如sunxin8086所写,获取正在运行的应用程序的一种方法是使用getRunningAppsProcesses()
方法。但是,条件info.importance == IMPORTANCE_FOREGROUND
无法唯一地确定当前应用。
确定当前前台应用程序的更好方法可能是检查processState
对象中的RunningAppProcessInfo
字段。此字段是隐藏字段,但您可以在RunningAppProcessInfo
类中查看。如果此值为ActivityManager.PROCESS_STATE_TOP
(也是
隐藏的静态常量),该过程是当前的前台进程。
例如代码是
final int PROCESS_STATE_TOP = 2;
ActivityManager.RunningAppProcessInfo currentInfo = null;
Field field = null;
try {
field = ActivityManager.RunningAppProcessInfo.class.getDeclaredField("processState");
} catch (Exception ignored) {
}
ActivityManager am = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
List<ActivityManager.RunningAppProcessInfo> appList = am.getRunningAppProcesses();
for (ActivityManager.RunningAppProcessInfo app : appList) {
if (app.importance == ActivityManager.RunningAppProcessInfo.IMPORTANCE_FOREGROUND
&& app.importanceReasonCode == ActivityManager.RunningAppProcessInfo.REASON_UNKNOWN) {
Integer state = null;
try {
state = field.getInt(app);
} catch (Exception e) {
}
if (state != null && state == PROCESS_STATE_TOP) {
currentInfo = app;
break;
}
}
}
return currentInfo;
注意:前Lolipop中不存在 processState
字段。在运行上述代码之前,请检查Build.VERSION.SDK_INT >= 21
。以上代码仅适用于Lollipop +。
Gaston的另一种方法(完全不同)和“当前应用”的含义与这种方法略有不同。
请为您选择一个。
[编辑]
Sam指出,我START_TASK_TO_FRONT
修改了PROCESS_STATE_TOP
。 (两个值均为2)
[EDIT2]
山姆有一个新的发现!要唯一地确定前台应用程序,还有一个 条件process.importanceReasonCode == 0
是必要的。上面的代码已更新。谢谢!
答案 2 :(得分:28)
这是获取Android L / Lollipop设备和Android M / Marshmallow设备当前顶级活动的精确解决方案。
首先调用这行代码:(一次)
Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS);
startActivity(intent);
上面的代码将打开一个名为“具有使用权限的应用”的屏幕。只需选中单选按钮打开/ true即可使用访问权限。 现在,在您的服务中或任何您想要的地方调用以下方法:
public void getTopActivtyFromLolipopOnwards() {
String topPackageName;
if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.LOLLIPOP) {
UsageStatsManager mUsageStatsManager = (UsageStatsManager) getSystemService(Context.USAGE_STATS_SERVICE);
long time = System.currentTimeMillis();
// We get usage stats for the last 10 seconds
List < UsageStats > stats = mUsageStatsManager.queryUsageStats(UsageStatsManager.INTERVAL_DAILY, time - 1000 * 10, time);
// Sort the stats by the last time used
if (stats != null) {
SortedMap < Long, UsageStats > mySortedMap = new TreeMap < Long, UsageStats > ();
for (UsageStats usageStats: stats) {
mySortedMap.put(usageStats.getLastTimeUsed(), usageStats);
}
if (mySortedMap != null && !mySortedMap.isEmpty()) {
topPackageName = mySortedMap.get(mySortedMap.lastKey()).getPackageName();
Log.e("TopPackage Name", topPackageName);
}
}
}
}
添加权限
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS"
tools:ignore="ProtectedPermissions" />
这将返回当前正在运行的活动的包名称,无论是facebook还是whatsapp。
此方法的唯一复杂因素是您需要提示用户允许应用使用统计信息...即第一步。
希望!这对每个人都有帮助。
答案 3 :(得分:5)
private String getProcess() throws Exception {
if (Build.VERSION.SDK_INT >= 21) {
return getProcessNew();
} else {
return getProcessOld();
}
}
//API 21 and above
private String getProcessNew() throws Exception {
String topPackageName = null;
UsageStatsManager usage = (UsageStatsManager) context.getSystemService(Constant.USAGE_STATS_SERVICE);
long time = System.currentTimeMillis();
List<UsageStats> stats = usage.queryUsageStats(UsageStatsManager.INTERVAL_DAILY, time - ONE_SECOND * 10, time);
if (stats != null) {
SortedMap<Long, UsageStats> runningTask = new TreeMap<Long,UsageStats>();
for (UsageStats usageStats : stats) {
runningTask.put(usageStats.getLastTimeUsed(), usageStats);
}
if (runningTask.isEmpty()) {
return null;
}
topPackageName = runningTask.get(runningTask.lastKey()).getPackageName();
}
return topPackageName;
}
//API below 21
@SuppressWarnings("deprecation")
private String getProcessOld() throws Exception {
String topPackageName = null;
ActivityManager activity = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
List<RunningTaskInfo> runningTask = activity.getRunningTasks(1);
if (runningTask != null) {
RunningTaskInfo taskTop = runningTask.get(0);
ComponentName componentTop = taskTop.topActivity;
topPackageName = componentTop.getPackageName();
}
return topPackageName;
}
//required permissions
<uses-permission android:name="android.permission.PACKAGE_USAGE_STATS"/>
<uses-permission android:name="android.permission.GET_TASKS"/>
答案 4 :(得分:3)
我认为不可能获得其他应用程序的任务,
这就是文档所说的
随着新的并发文件和活动的引入 即将发布的任务功能(参见Concurrent documents and activities in Recents screen below),. 现在不推荐使用ActivityManager.getRecentTasks()方法进行改进 用户隐私。为了向后兼容,此方法仍返回a 它的一小部分数据,包括调用应用程序自己的数据 任务以及可能的其他一些非敏感任务(例如Home)。如果 您的应用正在使用此方法检索自己的任务,使用 而不是android.app.ActivityManager.getAppTasks()来检索它 信息。
在此处查看Android L的api概述https://developer.android.com/preview/api-overview.html#Behaviors