Android PowerManager(1)

public void acquire(long timeout) {

synchronized (mToken) {

acquireLocked();

mHandler.postDelayed(mReleaser, timeout);

}

}

WakeLock是否已经acquire():

public boolean isHeld() {

synchronized (mToken) {

return mHeld;

}

}

WakeLock类型:

/** CPU:保持运转 屏幕:可以关闭 键盘灯:可以关闭 **/

public static final int PARTIAL_WAKE_LOCK = 0x00000001;

/** CPU:保持运转 屏幕:保持显示但可以是暗的 键盘灯:关闭 **/

@Deprecated

public static final int SCREEN_DIM_WAKE_LOCK = 0x00000006;

/** CPU:保持运转 屏幕:保持高亮 键盘灯:关闭 **/

@Deprecated

public static final int SCREEN_BRIGHT_WAKE_LOCK = 0x0000000a;

/** CPU:保持运转 屏幕:保持高亮 键盘灯:点亮 **/

@Deprecated

public static final int FULL_WAKE_LOCK = 0x0000001a;

注意事项:

1.普通锁:就是一点,不用时,必需及时release掉。

2.超时锁:这个看似很傻瓜,其实也有陷阱,如下两种方式:

private void acquireWakeLock1() {

wakeLock = powerManager.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK,

this.getClass().getCanonicalName());

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在如下情形下加锁时间的长短不同:

时间acquireWakeLock1acquireWakeLock2
00:00:00 runrun
00:01:00runrun
00:01:30 release
00:02:30releasegoTosleep()
... ...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;

最后

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

转存中…(img-0eYJINCM-1714932007876)]

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,需要这份全套学习资料的朋友可以戳我获取!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值