如何保持service长期在后台运行

android 一直运行的后台服务是不存在的,而且也不是最佳实践,因为一直运行的后台服务会耗费大量系统资源,影响其他程序的响应从而影响到用户体验。
可以考虑使用如下几种方案来达到一直运行的效果。

1. 调用startForeground方法,
android: Service vs SingleTop Activity moved to background
 Notification notification = new Notification(R.drawable.icon, getText(R.string.ticker_text),System.currentTimeMillis());
        Intent notificationIntent = new Intent(this, ExampleActivity.class);
        PendingIntent pendingIntent = PendingIntent.getActivity(this, 0, notificationIntent, 0);
        notification.setLatestEventInfo(this, getText(R.string.notification_title),getText(R.string.notification_message), pendingIntent);
        startForeground(ONGOING_NOTIFICATION_ID, notification);
将service从前台进程中移除:stopForeground()

Android中的进程是托管的,当系统进程空间紧张的时候,会依照优先级自动进行进程的回收。Android将进程分为6个等级,它们按优先级顺序由高到低依次是:

   1.前台进程( FOREGROUND_APP)
   2.可视进程(VISIBLE_APP )
   3. 次要服务进程(SECONDARY_SERVER )
   4.后台进程 (HIDDEN_APP)
   5.内容供应节点(CONTENT_PROVIDER)
   6.空进程(EMPTY_APP)

当service运行在低内存的环境时,将会kill掉一些存在的进程。因此进程的优先级将会很重要,可以使用startForeground 将service放到前台状态。这样在低内存时被kill的几率会低一些。


    这种方法在4.3以前都是可以任意用的。但是4.3系统优化android的安全机制,将一个service设置成前台进程,系统会在通知栏一直显示一个notifycation提示用户有一个service一直在执行。所以这种方法也不是完美的。

2、WakeLock机制
     WakeLock使用的场景:
     在使用一些产品列如微信、QQ之类的,如果有新消息来时,手机屏幕即使在锁屏状态下也会亮起并提示声音,这时用户就知道有新消息来临了。但是,一般情况下手机锁屏后,Android系统为了省电以及减少CPU消耗,在一段时间后会使系统进入休眠状态,这时,Android系统中CPU会保持在一个相对较低的功耗状态。针对前面的例子,收到新消息必定有网络请求,而网络请求是消耗CPU的操作,那么如何在锁屏状态乃至系统进入休眠后,仍然保持系统的网络状态以及通过程序唤醒手机呢?答案就是Android中的WakeLock机制。 

    获取锁:

WakeLock mWakeLock=null;
PowerManager pm = (PowerManager) getSystemService(Context.POWER_SERVICE);
/**
 * PowerManager.PARTIAL_WAKE_LOCK:保持CPU运转,屏幕和键盘灯可能是关闭的
 * PowerManager.SCREEN_DIM_WAKE_LOCK:保持CPU运转,运行屏幕显示但是屏幕有可能是灰的,允许关闭键盘灯
 * PowerManager.SCREEN_BRIGHT_WAKE_LOCK:保持CPU运转,屏幕高亮显示,允许关闭键盘灯
 * PowerManager.FULL_WAKE_LOCK:保持CPU运转,屏幕高亮显示,键盘灯高亮显示
 * PowerManager.ON_AFTER_RELEASE:当锁被释放时,保持屏幕亮起一段时间
 * PowerManager.ACQUIRE_CAUSES_WAKEUP:强制屏幕亮起
 */
mWakeLock = pm.newWakeLock(PowerManager.PARTIAL_WAKE_LOCK, "SoundRecorder");
mKeyguardManager = (KeyguardManager) getSystemService(Context.KEYGUARD_SERVICE);
mWakeLock.acquire();

          释放锁   
      if (mWakeLock.isHeld()) {
            mWakeLock.release();
        }
        最后注意权限问题。
还可以参考这篇博客:http://blog.csdn.net/ryantang03/article/details/8628753

3. 使用AlarmManager 发送定时任务 : Diamonds Are Forever. Services Are Not.
定期启动自己,相当于自己给自己的心跳包,这个因为只要执行代码,就托管给系统,就算进程挂掉,也会被系统定时启动,所以非常好。
AlarmManager是系统级的定时器,和TimerTask不一样。

 关于例子可以看:http://blog.csdn.net/ryantang03/article/details/9317499

4、 当service走ondestory的时候,发送一个自定义的广播,当收到广播的时候,重新启动service;
或者直接在service的ondestroy中startService
当使用类似口口管家等第三方应用或是在setting里-应用-强制停止时,APP进程可能就直接被干掉了,onDestroy方法都进不来,所以还是无法保证~.~

5.将service用新的进程开启,并且process name弄得跟系统进程一个名字,这样能防止很多三方杀进程的软件乱杀
 
6、开两个进程来相互监督,一旦其中一个进程被停止,另一个检测到后,立即或稍后重启另一个进程。
 
7、可以在onStartCommand中返回START_STICKY,这样系统有足够多资源的时候,就会重新开启service。

这篇博客讲得比较详细:http://blog.csdn.net/mad1989/article/details/22492519

你可能感兴趣的:(android)