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;

最后

分享一份工作1到5年以上的Android程序员架构进阶学习路线体系,希望能对那些还在从事Android开发却还不知道如何去提升自己的,还处于迷茫的朋友!

  • 阿里P7级Android架构师技术脑图;查漏补缺,体系化深入学习提升

  • **全套体系化高级架构视频;**七大主流技术模块,视频+源码+笔记

有任何问题,欢迎广大网友一起来交流
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!
脑图;查漏补缺,体系化深入学习提升

[外链图片转存中…(img-oBx4zUxS-1714399613786)]

  • **全套体系化高级架构视频;**七大主流技术模块,视频+源码+笔记

[外链图片转存中…(img-AvgFfZRd-1714399613788)]

有任何问题,欢迎广大网友一起来交流
《Android学习笔记总结+移动架构视频+大厂面试真题+项目实战源码》点击传送门,即可获取!

  • 23
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值