java - Android:获取当前打开的应用程序的栈(数组)

标签 java android

有什么方法可以获取当前正在运行或打开的android应用程序的列表,以便我可以找到设备中运行的最后一个应用程序。谢谢

最佳答案

这几天我一直在研究表情符号键盘。我花了三天的时间来获取前台应用程序包名称以发送表情符号图像,因为每当我想发送图像时,它都会弹出 intentchooser 以选择可以处理图像的应用程序之一 Actor 。

所有教程和链接都不适合我,因为谷歌弃用了用于获取当前正在运行的应用程序的 getRunningTasks() 方法。

然后我想到了使用 ResolveInfo 获取设备上当前正在运行的应用程序的堆栈,但它也没有奏效。

最后,我发现了一个在 API 21 中引入的名为 UsageStatsManager 的新类,它对我有用。 This github link提供如何使用此类获取正在运行的应用程序包名。

下面是我的代码如何让包名应用程序在顶部运行:

public class UStats {
    public static final String TAG = UStats.class.getSimpleName();
    @SuppressLint("SimpleDateFormat")
    private static final SimpleDateFormat dateFormat = new SimpleDateFormat("M-d-yyyy HH:mm:ss");

    public static ArrayList<String> printCurrentUsageStatus(Context context) {
        return printUsageStats(getUsageStatsList(context));
    }

    @TargetApi(Build.VERSION_CODES.LOLLIPOP)
    public static ArrayList<String> printUsageStats(List<UsageStats> usageStatsList) {
        HashMap<String, Integer> lastApp = new HashMap<String, Integer>();
        for (UsageStats u : usageStatsList) {
            lastApp.put(u.getPackageName(), (int) u.getLastTimeStamp());
            /*Log.d(TAG, "Pkg: " + u.getPackageName() + "\t" + "ForegroundTime: "
                    + u.getTotalTimeInForeground() + "\t" + "LastTimeStamp: " + new Date(u.getLastTimeStamp()));*/
        }
        Map<String, Integer> sortedMapAsc = sortByComparator(lastApp);
        ArrayList<String> firstApp = new ArrayList<>();
        for (Map.Entry<String, Integer> entry : sortedMapAsc.entrySet()) {
            String key = entry.getKey();
            //Integer value = entry.getValue();
            firstApp.add(key);
            /*System.out.println("package name: " + key + ", time " + new Date(Math.abs(value)));*/
        }

        return firstApp;
    }

    // To check the USAGE_STATS_SERVICE permission
    @TargetApi(Build.VERSION_CODES.LOLLIPOP)
    public static List<UsageStats> getUsageStatsList(Context context) {
        UsageStatsManager usm = getUsageStatsManager(context);
        Calendar calendar = Calendar.getInstance();
        long endTime = calendar.getTimeInMillis();
        calendar.add(Calendar.MINUTE, -1);
        long startTime = calendar.getTimeInMillis();

        Log.d(TAG, "Under getUsageStateList");
        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;
    }

    // Sort the map in the ascending order of the timeStamp
    private static Map<String, Integer> sortByComparator(Map<String, Integer> unsortMap) {
        List<Map.Entry<String, Integer>> list = new LinkedList<>(unsortMap.entrySet());

        // Sorting the list based on values
        Collections.sort(list, new Comparator<Map.Entry<String, Integer>>() {
            public int compare(Map.Entry<String, Integer> o1,
                               Map.Entry<String, Integer> o2) {
                return o2.getValue().compareTo(o1.getValue());
            }
        });

        // Maintaining insertion order with the help of LinkedList
        Map<String, Integer> sortedMap = new LinkedHashMap<String, Integer>();
        for (Map.Entry<String, Integer> entry : list) {
            sortedMap.put(entry.getKey(), entry.getValue());
        }
        return sortedMap;
    }

    @TargetApi(Build.VERSION_CODES.LOLLIPOP_MR1)
    @SuppressWarnings("ResourceType")
    private static UsageStatsManager getUsageStatsManager(Context context) {
        UsageStatsManager usm = (UsageStatsManager) context.getSystemService(Context.USAGE_STATS_SERVICE);
        return usm;
    }
}

然后我使用下面的代码简单地获取应用程序包名称的 ArrayList:

ArrayList<String> sortedApplication = UStats.printCurrentUsageStatus(SimpleIME.this);
Log.d("TAG", "applicationList: " + sortedApplication.toString());

不要忘记添加权限:

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

和下面的代码来检查我们的应用程序是否有权获取其他应用程序状态:

if (UStats.getUsageStatsList(this).isEmpty()) {
    Intent intent = new Intent(Settings.ACTION_USAGE_ACCESS_SETTINGS);
    Toast.makeText(MainActivity.this, "Enable Usage Access for YOUR_APP_NAME to use this app", Toast.LENGTH_LONG).show();
    startActivity(intent);
}

以上代码将打开一个访问设置页面,使我们的应用程序能够获取其他应用程序的使用情况统计信息(一次)。

希望这会有所帮助。

关于java - Android:获取当前打开的应用程序的栈(数组),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/38197811/

相关文章:

java - 杀死线程的替代方法

java - 如何从java中的try catch函数返回一个值?

java - 如何从 JNDI 变量初始化 web.xml 中的值?

android - 如何从自定义帐户检索联系方式?

android - Jquery mobile-如何在按钮单击时打开Android设置页面

java - 仅从文件夹创建 Eclipse 项目

java - 如何处理重复的类型参数?

android - Fabric/Crashlytics Android 为 Apikey 添加字符串资源导致 Crashlytics 开发者工具错误

android - 应用程序在 httprequest 上频繁崩溃

Android微调器在旋转时表现得很奇怪