NotificationListenerService 和 Doze 模式和 App Standby

Posted

技术标签:

【中文标题】NotificationListenerService 和 Doze 模式和 App Standby【英文标题】:NotificationListenerService and Doze mode and App Standby 【发布时间】:2016-08-22 09:07:36 【问题描述】:

我有一个应用程序可以监听电话通知并通过MessageApiandroid Wear 手表发送消息。除了一些搭载 Android 6 的设备,尤其是华为 Mate 8(看起来所有的华为 Android 6 都这样做)之外,一切都运行良好。

华为有自己的冻结应用后台处理(受保护的应用)的实现。从用户报告中,我确认我的应用在华为受保护的应用和 Android 6 的 Doze 模式下存在异常。该应用程序运行正常,但在显示关闭 15 分钟后,我的应用程序停止向连接的 Android Wear 手表发送消息。我的应用程序还可以记录收到的通知历史记录,15 分钟后什么都没有收到……直到手机显示屏打开并打开我的应用程序。之后,在手机显示屏关闭时应该到达的所有通知都会到达我的NotificationListenerService 实现并立即发送到手表。历史记录也证实了这一点。

任何想法如何为这些手机解决此问题,尤其是带有打瞌睡模式的 Android 6 的华为 Mate 8?

当设备处于打盹模式和/或应用处于待机模式时,NotificationListenerService 的正确行为是什么?

编辑

用户还确认他们的手机未处于省电模式,这也会影响后台应用程序及其服务。这个错误看起来像是华为独有的,因为没有 Nexus 用户报告过这个问题,而我的 OnePlus One with M 也没有这样做。 N 预览版也适用于 Nexus 设备。

编辑 2

我添加了一个可选的前台服务 (startForeground()),所以我的应用在通知中心有一个永久通知,因此我的应用应该被排除在每次电池优化之外。对于前台服务通知,我使用了NotificationCompat.PRIORITY_MIN 的优先级并添加了Notification.FLAG_ONGOING_EVENT 标志。这对华为手机有点帮助,但作用不大,现在延迟通知在屏幕打开后立即到达我的NotificationListenerService,而不是在打开我的应用程序后。我没有在我的NotificationListenerService 中使用startForeground(),而是在另一个Service 中使用,因为我无法控制它的生命周期。

【问题讨论】:

【参考方案1】:

对于华为设备(不确定是否适用于所有华为设备),您需要申请受保护的应用权限,以便您的应用在进入后台时不会被冻结。

检测华为设备是否具有受保护的应用权限:

private boolean hasProtectedAppsSetting() 
    Intent intent = new Intent();
    intent.setClassName("com.huawei.systemmanager", "com.huawei.systemmanager.optimize.process.ProtectActivity");

    List<ResolveInfo> list = getPackageManager().queryIntentActivities(intent,
            PackageManager.MATCH_DEFAULT_ONLY);
    return list.size() > 0;

打开华为受保护的应用设置页面:

private void showProtectedAppsSetting() 
    try 
        String cmd = "am start -n com.huawei.systemmanager/.optimize.process.ProtectActivity";
        if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN_MR1) 
            cmd += " --user " + getUserSerial();
        
        Runtime.getRuntime().exec(cmd);
     catch (IOException ignored) 
    


private String getUserSerial() 
    //noinspection ResourceType
    Object userManager = getSystemService("user");
    if (null == userManager) return "";

    try 
        Method myUserHandleMethod = android.os.Process.class.getMethod("myUserHandle", (Class<?>[]) null);
        Object myUserHandle = myUserHandleMethod.invoke(android.os.Process.class, (Object[]) null);
        Method getSerialNumberForUser = userManager.getClass().getMethod("getSerialNumberForUser", myUserHandle.getClass());
        long userSerial = (Long) getSerialNumberForUser.invoke(userManager, myUserHandle);
        return String.valueOf(userSerial);
     catch (NoSuchMethodException | IllegalArgumentException | InvocationTargetException | IllegalAccessException ignored) 
    
    return "";

很遗憾,我没有找到任何方法来检查用户是否已将您的应用授予受保护的应用。如果有人知道,请分享:)

参考: http://ndroid.info/ldquo_protected_appsrdquo_setting_on_huawei_phones_and_how_to_handle_it

【讨论】:

感谢您提供此信息。但是,正如问题中提到的,问题是即使在华为受保护的应用程序中给我的应用程序一个例外也不能解决“背景冻结”。甚至禁用 Android 6 的本机电池优化和前台服务。对于某些用户来说,为受保护的应用程序和电池优化添加一个例外就足够了,但不是全部。甚至大多数人都没有。【参考方案2】:

在看到您的帖子之前,我不知道 Android 的打盹模式。然后我读了this article,它似乎应该是这样工作的!如果您想在打盹模式下收到通知,请尝试使用PRIORITY_HIGHPRIORITY_MAX 作为您的通知优先级,但即使它有效,根据文章,它似乎也不是一个完整的解决方案。

【讨论】:

那篇文章是关于推送通知而不是本地通知。我的应用没有服务器,它完全离线,所以我无法通过高优先级推送通知唤醒手机。当手机显示通知时,也无法向手机发送推送通知...... 根据文章,是的,您将无法唤醒设备

以上是关于NotificationListenerService 和 Doze 模式和 App Standby的主要内容,如果未能解决你的问题,请参考以下文章