wakeLock.acquire(90 * 1000);
}
private void acquireWakeLock2() {
if(wakeLock == null){
wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,
this.getClass().getCanonicalName());
}
wakeLock.acquire(90 * 1000);
}
方法1和方法2在如下情形下加锁时间的长短不同:
时间 | acquireWakeLock1 | acquireWakeLock2 |
00:00:00 | run | run |
00:01:00 | run | run |
00:01:30 | release | |
00:02:30 | release | goTosleep() |
... ... | goTosleep() |
对于acquireWakeLock1(),每次执行都会重新锁定90s;
而对于acquireWakeLock2(),如果执行第二次的时候isHeld()为true,则进行第一次的锁定。
附:WakeLock未释放造成无法进入深度休眠(Suspend)时该如何定位:
抓取Log,搜索PowerManagerService关于WakeLock的打印,注意类似以下片段:
12-23 21:31:26.986 634 675 I PowerManagerService: Going to sleep by user request…
12-23 21:31:26.986 634 675 D PowerManagerNotifier: onGoToSleepStarted
12-23 21:31:26.986 634 675 D PowerManagerService: wakelock list dump: mLocks.size=2:
12-23 21:31:26.986 634 675 D PowerManagerService: No.0: PARTIAL_WAKE_LOCK 'com.zms.wakelock.WakeLockTestService’activated(flags=1, uid=10017, pid=787) total=2515481ms)
12-23 21:31:26.986 634 675 D PowerManagerService: No.1: PARTIAL_WAKE_LOCK 'AudioMix’activated(flags=1, uid=1013, pid=132) total=3306ms)
Log中可以看到有两个锁一直没有释放,所以无法进入深度休眠。
PowerManagerService有个ArrayList管理所有的WakeLock,为空时才能进入深度休眠:
// Table of all wake locks acquired by applications.
private final ArrayList mWakeLocks = new ArrayList();
以下是Log中打印处涉及的源码,打印出了WakeLock的类型,申请者的TAG,UID和PID,以及该锁acquire的时长,方便开发者定位检查:
private void dumpWakeLockLocked() {
final int numWakeLocks = mWakeLocks.size();
if (numWakeLocks > 0) {
Slog.d(TAG, “wakelock list dump: mLocks.size=” + numWakeLocks + “:”);
} else {
return;
}
for (int i = 0; i < numWakeLocks; i++) {
final WakeLock wakeLock = mWakeLocks.get(i);
String type = “”;
switch (wakeLock.mFlags & PowerManager.WAKE_LOCK_LEVEL_MASK) {
case PowerManager.PARTIAL_WAKE_LOCK:
type = “PARTIAL_WAKE_LOCK”;
break;
case PowerManager.FULL_WAKE_LOCK:
type = “FULL_WAKE_LOCK”;
break;
case PowerManager.SCREEN_BRIGHT_WAKE_LOCK:
type = “SCREEN_BRIGHT_WAKE_LOCK”;
break;
case PowerManager.SCREEN_DIM_WAKE_LOCK:
type = “SCREEN_DIM_WAKE_LOCK”;
break;
case PowerManager.PROXIMITY_SCREEN_OFF_WAKE_LOCK:
type = “PROXIMITY_SCREEN_OFF_WAKE_LOCK”;
break;
}
long total_time = SystemClock.uptimeMillis() - wakeLock.mActiveSince;
Slog.d(TAG, “No.” + i + “: " + type + " '” + wakeLock.mTag + “’”
-
“activated” + “(flags=” + wakeLock.mFlags
-
“, uid=” + wakeLock.mOwnerUid + “, pid=” + wakeLock.mOwnerPid + “)” + " total=" + total_time + “ms)”);
}
}
最后
其实Android开发的知识点就那么多,面试问来问去还是那么点东西。所以面试没有其他的诀窍,只看你对这些知识点准备的充分程度。so,出去面试时先看看自己复习到了哪个阶段就好。
上面分享的腾讯、头条、阿里、美团、字节跳动等公司2019-2021年的高频面试题,博主还把这些技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,上面只是以图片的形式给大家展示一部分。
Android学习PDF+学习视频+面试文档+知识点笔记
【Android思维脑图(技能树)】
知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。
【Android高级架构视频学习资源】
A2%E8%AF%95%E6%8B%BF%E9%AB%98%E8%96%AA%EF%BC%81.md)
【Android思维脑图(技能树)】
知识不体系?这里还有整理出来的Android进阶学习的思维脑图,给大家参考一个方向。
[外链图片转存中…(img-5PCC1Ujx-1646381517262)]
【Android高级架构视频学习资源】