Android Service 服务不被杀死的妙招

这篇文章主要介绍了Android Service 服务不被杀死的妙招的相关资料,需要的朋友可以参考下


Service是android 系统中的一种组件,它跟Activity的级别差不多,但是他不能自己运行,只能后台运行,并且可以和其他组件进行交互。

Android开发的过程中,每次调用startService(Intent)的时候,都会调用该Service对象的onStartCommand(Intent,int,int)方法,然后在onStartCommand方法中做一些处理。
从Android官方文档中,我们知道onStartCommand有4种int返回值,首先简单地讲讲int返回值的作用。

一、onStartCommand有4种返回值:

START_STICKY:如果service进程被kill掉,保留service的状态为开始状态,但不保留递送的intent对象。随后系统会尝试重新创建service,由于服务状态为开始状态,所以创建服务后一定会调用onStartCommand(Intent,int,int)方法。如果在此期间没有任何启动命令被传递到service,那么参数Intent将为null。
START_NOT_STICKY:“非粘性的”。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统不会自动重启该服务。

START_REDELIVER_INTENT:重传Intent。使用这个返回值时,如果在执行完onStartCommand后,服务被异常kill掉,系统会自动重启该服务,并将Intent的值传入。

START_STICKY_COMPATIBILITY:START_STICKY的兼容版本,但不保证服务被kill后一定能重启。

二、创建不被杀死的service

1.在service中重写下面的方法,这个方法有三个返回值, START_STICKY(或START_STICKY_COMPATIBILITY)是service被kill掉后自动重写创建

?
1
2
3
4
5
6
@Override
  public int onStartCommand(Intent intent, int flags, int startId)
  {
  return START_STICKY_COMPATIBILITY;
  //return super.onStartCommand(intent, flags, startId);
  }

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
@Override
  public int onStartCommand(Intent intent, int flags, int startId)
  {
  flags = START_STICKY;
  return super .onStartCommand(intent, flags, startId);
  // return START_REDELIVER_INTENT;
  }
@Override
public void onStart(Intent intent, int startId)
{
// 再次动态注册广播
IntentFilter localIntentFilter = new IntentFilter( "android.intent.action.USER_PRESENT" );
localIntentFilter.setPriority(Integer.MAX_VALUE); // 整形最大值
myReceiver searchReceiver = new myReceiver();
registerReceiver(searchReceiver, localIntentFilter);
super .onStart(intent, startId);
}

2.在Service的onDestroy()中重启Service.

?
1
2
3
4
5
6
public void onDestroy()
{
Intent localIntent = new Intent();
localIntent.setClass( this , MyService. class ); // 销毁时重新启动Service
this .startService(localIntent);
}

3.创建一个广播

?
1
2
3
4
5
6
7
8
public class myReceiver extends BroadcastReceiver
{
  @Override
  public void onReceive(Context context, Intent intent)
  {
  context.startService( new Intent(context, Google. class ));
  }
}

4.AndroidManifest.xml中注册广播myReceiver及MyService服务

?
1
2
3
4
5
6
7
8
9
10
11
<receiver android:name= ".myReceiver" >
       <intent-filter android:priority= "2147483647" ><!--优先级加最高-->
         <!-- 系统启动完成后会调用 -->
         <action android:name= "android.intent.action.BOOT_COMPLETED" />      
         <!-- 解锁完成后会调用 -->
         <action android:name= "android.intent.action.USER_PRESENT" />
         <!-- 监听情景切换 -->
         <action android:name= "android.media.RINGER_MODE_CHANGED" />      
       </intent-filter>
</receiver>
<service android:name= ".MyService" >

注:解锁,启动,切换场景激活广播需加权限,如启动完成,及手机机状态等。

?
1
2
<uses-permission android:name= "android.permission.RECEIVE_BOOT_COMPLETED" />
<uses-permission android:name= "android.permission.READ_PHONE_STATE" />

 亲测ZTE U795手机Android 4.0.4版本adb push到system\app下android:persistent="true"
变成核心程序,在360杀掉进程的时候,myReceiver照样有效,保证service重生。呃

KILL问题:

1. settings 中stop service

onDestroy方法中,调用startService进行Service的重启。

2.settings中force stop 应用

捕捉系统进行广播(action为android.intent.action.PACKAGE_RESTARTED)

3. 借助第三方应用kill掉running task

提升service的优先级,程序签名,或adb push到system\app下等

相较于/data/app下的应用,放在/system/app下的应用享受更多的特权,比如若在其Manifest.xml文件中设置persistent属性为true,则可使其免受out-of-memory killer的影响。如应用程序'Phone'的AndroidManifest.xml文件:

?
1
2
3
4
5
6
<application android:name= "PhoneApp"
        android:persistent= "true"
        android:label= "@string/dialerIconLabel"
        android:icon= "@drawable/ic_launcher_phone" >
    ...
</application>

设置后app提升为系统核心级别。


  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值