Android 开发之Service 探索如何保证Service不被杀死或被kill之后自动重启

前言:

在我司项目1.0版本的时候消息是使用的环信、用了之后发现各种bug,各种机型不支持导致app崩溃,于是在2.0版本果断去掉环信,使用了公众号用的那套消息系统(老大自己写的)并做了扩展升级。搞了近半个月终于是搞完了,项目也顺利上线......

你无法想象,检测未读消息/新消息我写了个线程,每隔30s去请求一个,好low的,app退出后你就拜拜了吧,肯定要改啊!用什么?service呗,于是开始service之旅...



废话连篇,开始我们的Service之旅吧!


1.我们要知道什么是Service?

A Service is an application component representing either an application's desire to perform a longer-running operation while not interacting with the user or to supply functionality for other applications to use. Each service class must have a corresponding  declaration in its package'sAndroidManifest.xml. Services can be started with Context.startService() and Context.bindService().

呵呵,你看得懂?

废话...

简单解释下就是:Service是一个应用程序组件,它能够在后台执行一些耗时较长的操作,并且不提供用户界面。服务能被其它应用程序的组件启动,即使用户切换到另外的应用时还能保持后台运行。此外,应用程序组件还能与服务绑定,并与服务进行交互,甚至能进行进程间通信(IPC)。 比如,服务可以处理网络传输、音乐播放、执行文件I/O、或者与content provider进行交互,所有这些都是后台进行的(抱歉,我抄的...)

附:Service官方介绍:传送门

  Android中文API:传送门


2.Service生命周期

Android 开发之Service 探索如何保证Service不被杀死或被kill之后自动重启_第1张图片Android 开发之Service 探索如何保证Service不被杀死或被kill之后自动重启_第2张图片


一会我们通过代码看结果,先了解...


3.Service基本类型(启动方式):

Started :通过应用程序组件(例如Activity)调用startService()启动服务:StartService(intent)系统通通过传入的intent搜索相关符合intent的Service,

依次执行其相关生命周期,service一旦启动就会一直在后台运行,直到调用stopService或stopSelf停止服务。

注:public void onStart(Intent intent, int startId) {}已过时,在2.0之后引入public int onStartCommand(Intent intent, int flags, int startId) {},flags,Service启动函数,后面介绍。

Bind:通过bindService()绑定服务,该提供了一个客户端/服务器接口,允许组建与服务进行交互、发送请求、返回结果,设置可以利用进程间通信夸进程执行这些操作;多个组件

可以同时与一个服务绑定,通过onUnbind()方法解绑服务,当所有组件解绑后,服务也被销毁。


接下来正式进入我们今天的话题:如何保证Service不被杀死或被kill之后自动重启!

1). onStartCommand() 返回常量Flag介绍:

  • START_STICKY 表示你希望系统可用的时候自动重启你的服务,但你不关心是否能获得最后一次的 Intent (例如,你可以重建自己的状态或者控制自己的 start/stop 生命周期)。
  • START_REDELIVER_INTENT 是为那些在被杀死之后重启时重新获得 Intent 的服务的,直到你用传递给 onStartCommand() 方法的 startId 参数调用 stopSelf() 为止。这里你会使用 Intent 和 startId 作为队列完成工作。
  • START_NOT_STICKY 用于那些杀掉也没关系的服务。这适合那些管理周期性任务的服务,它们只是等待下一个时间窗口工作。(摘自掘金
so,在内存不足服务被kill时,我们手动返回flag为START_STICKY / START_REDELIVER_INTENT(取决于重启是否需要重新获得intent),当内存足够时,服务会被重新创建.此方法然并卵,并不能使服务常驻...
@Override
public int onStartCommand(Intent intent, int flags, int startId) {
    Log.i("TAG", "Services onStartCommand");
    return START_REDELIVER_INTENT;
}

2).配置android:persistent="true"  ,persistent根据字面意思理解是持久...持久要持久!也就是常驻。但是通过测试发现被kill掉之后并不能重启...
3).查看其官方文档,有startForeground这个方法
  startForeground(int id, Notification notification)

Make this service run in the foreground, supplying the ongoing notification to be shown to the user while in this state.

其意思就是使服务在前台运行,发送一个通知给处于此状态的用户,前台必须提供一个状态栏通知
这里就涉及到service的进程优先级:当系统内存不足需要释放时,会按照优先级对进程回收,而android将进程分为六个等级
前台进程( FOREGROUND_APP)、可视进程(VISIBLE_APP )、次要服务进程(SECONDARY_SERVER )
后台进程 (HIDDEN_APP)、内容供应节点(CONTENT_PROVIDER)、空进程(EMPTY_APP)

可在service onStartCommand()方法中如此操作:
NotificationCompat.Builder builder = new NotificationCompat.Builder(G.applicationContext);
Notification notification = builder.build();
notification.flags = Notification.FLAG_FOREGROUND_SERVICE;
startForeground(0, notification);
Log.i("Service", "UnreadMessageServices onStartCommand");
return START_STICKY;
写一个状态通知栏大家都会吧,这里就不详说了,不会的自行google...
运行后效果为 如图:
Android 开发之Service 探索如何保证Service不被杀死或被kill之后自动重启_第3张图片
我们将service置为前台服务时,一般情况还是不想让用户看到对吧。startForeground()方法。此方法有两个参数:唯一标识通知的整数值、状态栏通知Notification对象。当我传id不为0时,Notification会显示,当id=0时则不显示到通知栏,原因嘛...送上传送门Android Service startForeground() 不显示Notification问题)。
这样做只是在低内存是降低该service被kill掉的几率,并不能真正使service常驻。
4).还有方法是说让其成为系统应用...  好吧这个我确实没测试,也不想..听说apk无法卸载;
设置该service为独立进程,貌似提升了优先级,但是照样被kill...  pass ;
在onDestory()方法中重启service,能不能不用这么low的方法...我没做测试


这几种方法只能提升service优先级/存活率,但是还不能解决其被杀毒软件强行kill的命运...

我的解决方案:
1.首先设置服务为开机自启
public class BootBroadcastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        Intent unreadCountService = new Intent(context, UnreadCountService.class);
        Intent checkService = new Intent(context, CheckService.class);
        context.startService(unreadCountService);
        context.startService(checkService);
        Log.v("TAG", "开机自动服务自动启动.....");
    }
}
同时需要配置其权限AndroidManifest.xml

android:name="android.permission.RECEIVE_BOOT_COMPLETED" />

    android:name=".help.BootBroadcastReceiver"
    android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
    
        android:name="android.intent.action.BOOT_COMPLETED" />
        android:name="android.intent.category.DEFAULT" />
    

2.利用系统广播  Intent.ACTION_TIME_TICK 每隔一分钟检测一次Service的运行状态
private class CheckBroadCastReceiver extends BroadcastReceiver {
    @Override
    public void onReceive(Context context, Intent intent) {
        if (intent.getAction().equals("Intent.ACTION_TIME_TICK")) {
            //检查Service状态
            if (isServiceRunning(context, "rain.myapp.help.UnreadCountService") == false) {
                //重启服务
                Intent i = new Intent(context,UnCountService.class);
                context.startService(i);
            }
        }
    }
}

/**
 * 判断服务是否正在运行中
 *
 * @param context     Context对象
 * @param serviceName Service全名
 * @return
 */
private boolean isServiceRunning(Context context, String serviceName) {
    if (!TextUtils.isEmpty(serviceName) && context != null) {
        ActivityManager activityManager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE);
        ArrayList runningServiceInfoList
                = (ArrayList) activityManager.getRunningServices(100);
        for (Iterator iterator = runningServiceInfoList.iterator(); iterator.hasNext(); ) {
            ActivityManager.RunningServiceInfo runningServiceInfo = iterator.next();
            if (serviceName.equals(runningServiceInfo.service.getClassName().toString()))
                return true;
        }
    } else return false;
    return false;
}

公司项目该功能做到这一步就已经足够了,暂时也不打算深入研究了。作为Android开发者,我们本身是有必要去维护Android的生态环境而不是一昧的去破坏... 到此为止!菜鸟写博客,诸多不合之处欢迎指出,还望无喷...

你可能感兴趣的:(Android)