美文网首页Android原理程序员Android Developer
Android 开发之Service 探索如何保证Service

Android 开发之Service 探索如何保证Service

作者: 老鱼的储物柜 | 来源:发表于2016-12-05 10:45 被阅读2560次

    前言:

    在我司项目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 correspondingdeclaration in its package'sAndroidManifest.xml. Services can be started withContext.startService()andContext.bindService().

    呵呵,你看得懂?

    废话...

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

    附:Service官方介绍:传送门

    Android中文API:传送门

    2.Service生命周期

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

    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),当内存足够时,服务会被重新创建.此方法然并卵,并不能使服务常驻...

    @Overridepublic intonStartCommand(Intent intent, intflags, intstartId) {    Log.i("TAG","Services onStartCommand");returnSTART_REDELIVER_INTENT;}

    2).配置android:persistent="true"  ,persistent根据字面意思理解是持久...持久要持久!也就是常驻。但是通过测试发现被kill掉之后并不能重启...

    3).查看其官方文档,有startForeground这个方法

    startForeground(int id,Notificationnotification)

    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 =newNotificationCompat.Builder(G.applicationContext);Notification notification = builder.build();notification.flags= Notification.FLAG_FOREGROUND_SERVICE;startForeground(0,notification);Log.i("Service","UnreadMessageServices onStartCommand");returnSTART_STICKY;

    写一个状态通知栏大家都会吧,这里就不详说了,不会的自行google...

    运行后效果为 如图:

    我们将service置为前台服务时,一般情况还是不想让用户看到对吧。startForeground()方法。此方法有两个参数:唯一标识通知的整数值、状态栏通知Notification对象。当我传id不为0时,Notification会显示,当id=0时则不显示到通知栏,原因嘛...送上传送门(Android ServicestartForeground() 不显示Notification问题)。

    这样做只是在低内存是降低该service被kill掉的几率,并不能真正使service常驻。

    4).还有方法是说让其成为系统应用...  好吧这个我确实没测试,也不想..听说apk无法卸载;

    设置该service为独立进程,貌似提升了优先级,但是照样被kill...  pass ;

    在onDestory()方法中重启service,能不能不用这么low的方法...我没做测试

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

    我的解决方案:

    1.首先设置服务为开机自启

    public classBootBroadcastReceiverextendsBroadcastReceiver {

    @Override

    public voidonReceive(Context context,Intent intent) {

    Intent unreadCountService =newIntent(context,UnreadCountService.class);

    context.startService(unreadCountService);

    Log.d("MessageService","开机服务自启...");

    }}

    同时需要配置其权限AndroidManifest.xml

    2.利用系统广播Intent.ACTION_TIME_TICK 每隔一分钟检测一次Service的运行状态

    public classCheckBroadCastReceiverextendsBroadcastReceiver {

    @Override

    public voidonReceive(Context context,Intent intent) {

    if(intent.getAction().equals(Intent.ACTION_TIME_TICK)) {

    //检查Service状态

    Log.e("MessageService","onReceive: "+"检测MessageService是否运行 :"+

    UtilsHelper.isServiceRunning(context,"com.dailylifeapp.app.and.dailylife.helper.UnreadCountService"));

    if(UtilsHelper.isServiceRunning(context,"com.dailylifeapp.app.and.dailylife.helper.UnreadCountService") ==false) {

    //重启服务

    Intent i =newIntent(context,UnreadCountService.class);

    context.startService(i);

    }}}}

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

    相关文章

      网友评论

      本文标题:Android 开发之Service 探索如何保证Service

      本文链接:https://www.haomeiwen.com/subject/ukykmttx.html