NotificationListenerService和Doze模式以及应用待机模式

9
我有一个应用程序可以监听电话通知并通过MessageApi向Android Wear手表发送消息。除了一些安卓6设备(尤其是华为 Mate 8,貌似所有华为安卓6的设备都会这样),一切正常。
华为有自己的冻结应用程序后台处理的实现(受保护的应用程序)。从用户报告中,我确认我的应用程序在华为的保护应用程序和安卓6的Doze模式下有一个例外。应用程序正常工作,但在关闭显示15分钟后,我的应用程序就停止向连接的Android Wear手表发送消息。我的应用程序还可以记录接收到的通知历史记录,而在15分钟内没有任何通知到达......直到手机的显示屏亮起并且我的应用程序打开。之后,所有应该在手机的显示屏关闭期间到达的通知都进入我的NotificationListenerService实现并一次性发送到手表。这也通过记录的历史得到了证实。
有什么想法可以为这些手机解决这个问题,特别是具有Doze模式的华为Mate 8?
当设备处于Doze模式和/或应用程序处于待机模式时,NotificationListenerService的正确行为是什么?
编辑
用户还确认他们的手机没有处于省电模式,这也会影响后台应用程序及其服务。这个错误看起来是华为专属的,因为没有Nexus用户报告此问题,我的OnePlus One与M也没有这样做。而且,在N预览版中,Nexus设备工作得很好。
编辑2
我已添加了可选前台服务(startForeground()),因此我的应用程序在通知中心具有永久通知,因此我的应用程序应该从每个电池优化中排除。对于前台服务通知,我使用了NotificationCompat.PRIORITY_MIN的优先级,并添加了Notification.FLAG_ONGOING_EVENT标志。这在华为手机上有所帮助,但作用不大,现在延迟的通知在打开屏幕后立即进入我的NotificationListenerService,而不是在打开我的应用程序后。我没有在我的NotificationListenerService中使用startForeground(),而是在另一个Service中使用,因为我无法控制它的生命周期。
2个回答

2
对于华为设备(不确定是否适用于所有华为设备),您需要请求受保护的应用程序权限,以便在您的应用程序进入后台时不会被冻结。
要检测华为设备是否具有受保护的应用程序权限:
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 "";
}

很遗憾,我还没有找到任何方法来检查用户是否将您的应用程序作为受保护的应用程序授权。如果有人知道,请分享一下 :)

Reference: http://ndroid.info/ldquo_protected_appsrdquo_setting_on_huawei_phones_and_how_to_handle_it


感谢您提供这些信息。但是,正如问题所述,即使在华为受保护的应用程序中给我的应用程序设置了异常情况,也不能解决“后台冻结”问题。即使同时禁用Android 6原生电池优化和前台服务,对于某些用户来说,只需添加受保护的应用程序和电池优化的异常情况就足够了,但并非所有用户都是如此。甚至不是大多数用户。 - shelll

0

在看到您的帖子之前,我不知道Android的Doze模式。然后我读了this article,看起来这就是它应该工作的方式!如果你想在Doze模式下收到通知,请尝试使用PRIORITY_HIGH或者PRIORITY_MAX作为你的通知优先级,但即使它能够工作,根据文章的说法它并不是一个完整的解决方案。


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

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接