前言:
在我司项目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的生态环境而不是一昧的去破坏... 到此为止!菜鸟写博客,诸多不合之处欢迎指出,还望无喷...