华为 EMUI 4.0+ 上的电池优化(唤醒锁)

Battery optimizations (wakelocks) on Huawei EMUI 4.0+

美好的一天, 情况:

我正在开发用作运动 tracker/navigation 应用程序的 Android 应用程序 - 因此它需要永久连接到 GPS 以及永久唤醒设备。每秒录制一次。

目前的解决方案多年来一直有效,这要归功于 wakelocks 让设备保持唤醒状态。

打瞌睡模式在Android 6.0+ 复杂的情况下,但不是这种情况。

在华为设备上可能是不同类型的优化。

这是日志的一部分:

10-10 10:33:18.462 1014-384 D/PFW.HwPFWAppWakeLockPolicy: getUidWakeLock uid: 10097 wakelock >= 10 mins 10-10 10:33:18.543 1014-384 D/PFW.HwPFWAppWakeLockPolicy: force stop abnormal wakelock app uid: 10097 10-10 10:33:18.558 1014-384 I/ActivityManager: Force stopping menion.android.locus appid=10097 user=0: from pid 1014

所以大约。超过 30 分钟,系统简单地决定应用程序使用过多 wakelocks 并完全终止它的所有服务,历史记录,简单地 kill.

对此行为有任何经验和建议,如何更好地完成这个简单的任务(屏幕关闭时永久记录 GPS 位置)?

正如我在开始时所写,在除新华为之外的所有其他设备上,这样的系统可以正常运行多年。

编辑:一位用户评论后的备注(已删除?),华为电池管理器中的"whitelist"应用程序(标记为"protected application")无效关于这个问题。

您是否尝试过设置一个定期释放唤醒锁然后在几秒钟后重新获取它的警报?如果问题是华为的 Android 不喜欢滥用唤醒锁,你偶尔发布一下也许他们就没事了? 例如:我想你会有一个后台进程 运行 在前台。如果是这样,请在您的 onStartCommand 中插入:

setupWakeupAlarm(context);

其中方法定义为:

private static void setupWakeupAlarm(Context context) {
    AlarmManager mWakeUpAlarmManager = (AlarmManager) context.getSystemService(Context.ALARM_SERVICE);
    PendingIntent mWakeUpAlarmIntent;
    Intent serviceIntent;
    serviceIntent = new Intent(context, SSAlarmReceiver.class);
    mWakeUpAlarmIntent = PendingIntent.getBroadcast(context, 0, serviceIntent, PendingIntent.FLAG_UPDATE_CURRENT);
    // every 5 minutes 
    mWakeUpAlarmManager.setExact(AlarmManager.ELAPSED_REALTIME_WAKEUP,
            SystemClock.elapsedRealtime() + WAKEUP_ALARM_FREQUENCY, mWakeUpAlarmIntent);
    Log.d("TAG", "wakup alarm set up or reset!");
}

接收者是本地人class:

static public class SSAlarmReceiver extends WakefulBroadcastReceiver {
    @Override
    public void onReceive(final Context context, Intent intent) {
        setupWakeupAlarm(context);
        mBackgroundService.stopForeground(true);
        if (mWakeLock.isHeld())
            mWakeLock.release();
        new Timer().schedule(
                new TimerTask() {
                    @Override
                    public void run() {
                        mWakeLock.acquire();
                        mBackgroundService.startForeground(mNotificationId, mNotification.getNotification());
                        }
                },
                3000
        );
    }
}

请注意,在我的例子中,我的后台服务 运行 也在前台,我决定停止前台。不确定是否有必要。

风险当然是在唤醒锁未激活的 3 秒内,您的进程可能会被终止。

有两个华为系统应用程序可能会终止用户应用程序以节省电量:

  • SystemManager (com.huawei.systemmanager) 会终止所有在屏幕关闭后仍然 运行 的应用程序,除非它们在 "Protected Apps" 列表中。
  • PowerGenie (com.huawei.powergenie) 会杀死任何长时间持有唤醒锁的应用程序。

听起来您的应用正在被 PowerGenie 杀死。您可以利用 PowerGenie 的硬编码唤醒锁定标签白名单来避免这种情况。例如,如果您的唤醒锁的标签是 "LocationManagerService",PowerGenie 将忽略它,因为系统服务持有具有相同标签的唤醒锁并且 PowerGenie 已将其列入白名单。