android 锁屏的防进程被杀

花了半天时间研究下了自定义锁屏,发现其实实现并不是很神秘。不过有些地方还是值得注意。

 想在程序中监听屏幕SCREEN_ON和SCREEN_OFF这两个action,实现屏幕锁定状态的监听,从而实现自己的相应功能。比较奇怪的是这两个action只能通过代码的形式注册才能被监听到,在AndroidManifest.xml中注册根本监听不到。去网上查了一下,原来是PowerManager那边在发这个广播的时候做了限制,限制只有register到代码中的receiver才能接收。特此记录!

首先说流程,锁屏界面一般是在关闭屏幕时启用,打开屏幕时展现在我们面前,所以我们知道了锁屏的时机,恰好屏幕开关会发出相应的广播,所以我们也可以像系统一样捕获到屏幕开关的事件。

广播对应的两个action

 

引用

 

android.intent.action.SCREEN_ON

android.intent.action.SCREEN_OFF

 

 

有趣的是我在我的me860上还看到了两个广播

 

引用

 

android.intent.action.batteryprofile.SCR_OFF

android.intent.action.batteryprofile.SCR_ON

 

 

这可能是moto自己定义的广播,看大家需要了。

如果单单只截获到广播,锁屏的界面怎么替换系统锁屏呢?其实我们这里并没有替换掉系统锁屏,只是我们自己的锁屏开启了系统锁屏,从而达到了一个替换的效果。

这里还是要注意点,上述action不能在AndroidManifest.xml中注册,不能触发。所以这里我采用的方法是放在一个service中动态注册,截获广播正常。

这里还顺带提下service防杀的功能,因为service的进程如果不存在了,那么屏幕开关的广播是无论如何也接收不到的。

如果是系统回收掉的service,系统会在过段时间,资源充足的情况下再启动起来,不过我们可不想自己的service坐冷板凳,所以对付系统关闭service我们采取如下方法

在service生命周期中

Java代码 

Intent startIntent=null;  

@Override 

    public void onStart(Intent intent, int startId) {  

        startIntent=intent;  

        IntentFilter filter=new IntentFilter();  

        filter.addAction(Intent.ACTION_SCREEN_ON);  

        filter.addAction(Intent.ACTION_SCREEN_OFF);  

        receiver=new RelativeBroadcastReceiver();  

        registerReceiver(receiver, filter);  

        System.out.println("service onStart and action is "+intent.getAction());  

        System.out.println("service onStart and startId is "+startId);  

    };  

      

    @Override 

    public void onDestroy() {  

        // TODO Auto-generated method stub  

        System.out.println("service onDestroy");  

        unregisterReceiver(receiver);  

        if(startIntent!=null){  

            System.out.println("serviceIntent not null");  

            startService(startIntent);  

        } 

 

Intent startIntent=null;

@Override

       public void onStart(Intent intent, int startId) {

              startIntent=intent;

              IntentFilter filter=new IntentFilter();

              filter.addAction(Intent.ACTION_SCREEN_ON);

              filter.addAction(Intent.ACTION_SCREEN_OFF);

              receiver=new RelativeBroadcastReceiver();

              registerReceiver(receiver, filter);

              System.out.println("service onStart and action is "+intent.getAction());

              System.out.println("service onStart and startId is "+startId);

       };

      

       @Override

       public void onDestroy() {

              // TODO Auto-generated method stub

              System.out.println("service onDestroy");

              unregisterReceiver(receiver);

              if(startIntent!=null){

                     System.out.println("serviceIntent not null");

                     startService(startIntent);

              }

 

 

我们保留了开启service的intent,当进入ondestroy周期中时再启动一次自己,系统看到你这劲头也只好答应你常驻内存了。

 

不过如果是一些内存管理软件杀掉了程序进程,以上方法就没用了,我们还是可以通过截获一些关键广播来启动自己的service,类似91助手等就是这样。可以监听wifi连接,电池电量发生变化等广播来启动自己的service。

 

保证自己的service常驻,这时候就该我们自己的广播接收器发挥作用了。

Java代码 

@Override 

    public void onReceive(Context context, Intent intent) {  

        // TODO Auto-generated method stub  

        String action=intent.getAction();  

        System.out.println("action is "+action);  

        Intent lockIntent=new Intent(context,MyLockScreen.class);   

        lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);  

        context.startActivity(lockIntent);  

 

@Override

       public void onReceive(Context context, Intent intent) {

              // TODO Auto-generated method stub

              String action=intent.getAction();

              System.out.println("action is "+action);

              Intent lockIntent=new Intent(context,MyLockScreen.class);

              lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);

              context.startActivity(lockIntent);

}

 

需要加上lockIntent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);没有加的时候系统会报错,但是加上以后也有问题,这会导致多次退出才能退出自定义的锁屏界面。

其实可以自定义一个stack来管理这些activity,有其他办法的同学请提示我改正。

 

启动了一个activity以后我们发现还是原来的锁屏界面,这也是前面提到的,我们的锁屏需要打开系统锁屏。

打开系统锁屏:

Java代码 

super.onCreate(savedInstanceState);   

        getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);  

        getWindow().addFlags(WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);  

        setContentView(R.layout.main); 

 

super.onCreate(savedInstanceState);

              getWindow().addFlags(WindowManager.LayoutParams.FLAG_SHOW_WHEN_LOCKED);

              getWindow().addFlags(WindowManager.LayoutParams.FLAG_DISMISS_KEYGUARD);

              setContentView(R.layout.main);

 

添加了这两个flag以后,系统锁屏就被替换成了我们自己的锁屏界面。

锁屏替换功能也就完成了。其实还可以修改系统锁屏,不过设计到框架层的修改,推广也很麻烦,所以掠过不说了

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值