Android-WakeLock(唤醒锁与CPU休眠/屏幕常亮)
参考: https://blog.csdn.net/wh_19910525/article/details/8287202 http://landerlyoung.github.io/blog/2014/10/31/androidzhong-de-wakelockshi-yong/
一.使用
PowerManager pm = (PowerManager) context.getSystemService(Context.POWER_SERVICE);
PowerManager.WakeLock wakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "MyTAG");
wakeLock.acquire();
在Manifest中添加如下权限
<uses-permission android:name="android.permission.WAKE_LOCK" />
WakeLock级别:
PARTIAL_WAKE_LOCK 保持CPU运转,屏幕和键盘背光可能关闭
SCREEN_DIM_WAKE_LOCK 保持CPU运转,保持屏幕常亮(亮度低),键盘背光可能关闭
SCREEN_BRIGHT_WAKE_LOCK 保持CPU运转,保持屏幕和键盘背光高亮
FULL_WAKE_LOCK 保持CPU运转,保持屏幕和键盘背光高亮(亮度最高)
ACQUIRE_CAUSES_WAKEUP 强制亮屏,针对一些必须通知用户的操作
ON_AFTER_RELEASE 当锁被释放时,保持亮屏一段时间(如果释放时屏幕没亮,则不会亮屏)
PROXIMITY_SCREEN_OFF_WAKE_LOCK 和接近传感器配合,当用户接近屏幕时黑屏,离开时亮屏(例如打电话),该API在API21后开放,以前被hide
保持屏幕长亮的WakeLock被建议弃用,系统推荐如下方法(当Activity或view可见时,屏幕才保持常亮):
在Activity.onCreate()中: getWindow().addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON);
或在xml布局中: android:keepScreenOn="true"
或对View设置: view.setKeepScreenOn(true);
FLAG_KEEP_SCREEN_ON实际就是一个SCREEN_BRIGHT_WAKE_LOCK级别的WakeLock
创建和释放锁都由系统自动管理,更加方便和安全,在后面通过adb命令验证
屏幕相关的其它FLAG:
WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD 解锁屏幕
WindowManager.LayoutParams.FLAG_TURN_SCREEN_ON 点亮屏幕
WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED 屏幕锁定时也能显示
WindowManager.LayoutParams.FLAG_ALLOW_LOCK_WHILE_SCREEN_ON 屏幕打开时允许锁屏
屏幕锁屏和解锁(需要权限"android.permission.DISABLE_KEYGUARD")
KeyguardManager keyguardManager = (KeyguardManager) context.getSystemService(KEYGUARD_SERVICE);
KeyguardManager.KeyguardLock keyguardLock = keyguardManager.newKeyguardLock("unLock");
keyguardLock.reenableKeyguard(); // 锁屏
//keyguardLock.disableKeyguard();// 解锁
!注意:
如果申请partial wakelock,那么即使用户按Power键,系统也不会Sleep
如果申请其它wakelocks,用户按Power键,系统还是会Sleep
PowerManager方法
boolean isScreenOn() 判断屏幕是否亮着,在API20被弃用,推荐isInteractive()
void goToSleep(long time) 强制休眠(系统APP权限"android.permission.DEVICE_POWER",按下电源键锁屏时调用该方法)
void wakeUp(long time) 强制唤醒(权限同上,按下电源键时调用该方法)
void reboot(String reason) 重启手机(系统APP权限"android.permission.REBOOT"),参数:"recovery","fastboot"
WakeLock方法:
void acquire() 获得WakeLock,除非显式释放,否则不会解锁(经测试,APP进程被杀死后,锁会失效)
void acquire(long timeOut) 当超过timeOut之后系统自动释放WakeLock
void release() 释放WakeLock
boolean isHeld() 是否已经获取WakeLock
void setReferenceCounted(boolean value) 是否使用引用计数(默认ture): 一个WakeLock调用acquire()多次,也必须release()多次才能释放,
如果释放次数比acquire()多,则抛出异常: java.lang.RuntimeException: WakeLock under-locked MyTAG
二.原理
通过adb命令查看WakeLock锁的个数:
adb shell dumpsys power
Wake Locks: size=4
mLock:346803 PARTIAL_WAKE_LOCK 'MyTAG' ACQ=-1s175ms (uid=10201 pid=3862)
mLock:141928 SCREEN_DIM_WAKE_LOCK 'MyTAG' ACQ=-1s174ms (uid=10201 pid=3862)
mLock:402427 SCREEN_BRIGHT_WAKE_LOCK 'MyTAG' ACQ=-1s173ms (uid=10201 pid=3862)
mLock:124789 SCREEN_BRIGHT_WAKE_LOCK 'WindowManager' ON_AFTER_RELEASE ACQ=-1s70ms (uid=1000 pid=1196 ws=WorkSource{10201})
注:最后一个Lock是由FLAG_KEEP_SCREEN_ON创建的,由系统用户进程管理(uid=1000是系统用户)
在创建了 PowerManager.WakeLock 后,有两种机制,第一种是不计数锁机制,另一种是计数锁机制。
可以通过 setReferenceCounted(boolean value) 来指定,一般默认为计数机制。
这两种机制的区别在于,前者无论 acquire() 了多少次,只要通过一次 release()即可解锁。
而后者正真解锁是在( --count == 0 )的时候,同样当 (count == 0) 的时候才会去申请加锁。
所以 PowerManager.WakeLock 的计数机制并不是正真意义上的对每次请求进行申请/释放每一把锁,
它只是对同一把锁被申请/释放的次数进行了统计,然后再去操作。
Wakelock框架的大致流程:
1).应用层 /frameworks/base/core/java/android/os/PowerManager.java
WakeLock.acquire() -> PowerManagerService.acquireWakeLock()
2).frameworks层 /frameworks/base/services/java/com/android/server/PowerManagerService.java
管理所有APP申请的wakelock,比如音视、频播放器、camera等
static final String PARTIAL_NAME ="PowerManagerService";
public static native void nativeAcquireWakeLock(int lock, String id);
nativeAcquireWakeLock(PARTIAL_WAKE_LOCK_ID, PARTIAL_NAME);
3).JNI层 /frameworks/base/core/jni/android_os_Power.cpp
static void acquireWakeLock(JNIEnv *env, jobject clazz, jint lock, jstring idObj){
...
const char *id = env->GetStringUTFChars(idObj, NULL);
acquire_wake_lock(lock, id);
env->ReleaseStringUTFChars(idObj, id);
}
4).kernel层 /hardware/libhardware_legacy/power/power.c
int acquire_wake_lock(int lock, const char* id){
...
return write(fd, id, strlen(id));
}
fd是文件描述符: "/sys/power/wake_lock"
id是从frameworks层传来的参数: "PowerManagerService"
简书: https://www.jianshu.com/p/2cfd179ef8dc
CSDN: https://blog.csdn.net/qq_32115439/article/details/80169222
GitHub博客: http://lioil.win/2018/05/02/Android-WakeLock.html
Coding博客: http://c.lioil.win/2018/05/02/Android-WakeLock.html