Android中如何做到Service被关闭后又自动启动

*07 Android 同时被 2 个专栏收录
76 篇文章 0 订阅
14 篇文章 0 订阅
 

首先要说的是,用户可能把这种做法视为流氓软件。大部分时候,程序员也不想把软件做成流氓软件,没办法,领导说了算。

 

我们在使用某些Android应用的时候,可能会发现安装了某应用以后,会有一些服务也会随之运行。而且,这些服务每次都会随着手机开机而启动。有的服务做的更绝,当用户在运行的服务中手动停止该服务以后,过了一段时间,服务又自动运行了。虽然,从用户的角度来说,这种方式比较流氓。但是,从程序员的角度来说,这是如何做到的呢?经过研究,我发现有一种方式是可以实现的。下面就和大家分享。

 

先简单介绍,一会儿会贴上全部代码。

 

如何做到开机启动?

 

这个比较简单,网上的资料够多,只要实现一个BroadcastReceiver,监听手机启动完成的事件ACTION_BOOT_COMPLETED即可。需要注意的是,好像不能用模拟器,要用手机测试。

 

那如何做到启动一个Service,并且在用户关闭后能自动又启动了呢?

 

一般的,都会在上面说到的BroadcastReceiver的实现里面,监听手机启动完成后,启动一个Service,这是一般的做法。问题是,用户能够在服务里看到这个Service是常驻的。如果用户很敏感,就会停止该Service,甚至直接卸载掉相关的应用。那么,怎样才能定期实现某功能,又不让用户直接看到这个Service呢?聪明的你一定立即就想到了,如果不直接启动Service,而是启动一个timmer,或者alarmManager,然后每隔一段时间去启动Service,做完事情以后关闭掉Service就可以了。

 

 

还是看下面的全部代码吧,不过多解释了。这些代码中还是有不少概念的,不熟悉AlarmManager、PendingIntent、BroadcastReceiver、Service等等这些概念的同学可以百度一下。

 

  1. package com.arui.framework.android.daemonservice;
  2. import android.app.AlarmManager;
  3. import android.app.PendingIntent;
  4. import android.content.BroadcastReceiver;
  5. import android.content.Context;
  6. import android.content.Intent;
  7. import android.os.SystemClock;
  8. public class BootBroadcast extends BroadcastReceiver {
  9. @Override
  10. public void onReceive(Context context, Intent mintent) {
  11. if (Intent.ACTION_BOOT_COMPLETED.equals(mintent.getAction())) {
  12. // 启动完成
  13. Intent intent = new Intent(context, Alarmreceiver.class);
  14. intent.setAction("arui.alarm.action");
  15. PendingIntent sender = PendingIntent.getBroadcast(context, 0,
  16. intent, 0);
  17. long firstime = SystemClock.elapsedRealtime();
  18. AlarmManager am = (AlarmManager) context
  19. .getSystemService(Context.ALARM_SERVICE);
  20. // 10秒一个周期,不停的发送广播
  21. am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstime,
  22. 10 * 1000, sender);
  23. }
  24. }
  25. }


 

  1. package com.arui.framework.android.daemonservice;
  2. import android.content.BroadcastReceiver;
  3. import android.content.Context;
  4. import android.content.Intent;
  5. public class Alarmreceiver extends BroadcastReceiver {
  6. @Override
  7. public void onReceive(Context context, Intent intent) {
  8. if (intent.getAction().equals("arui.alarm.action")) {
  9. Intent i = new Intent();
  10. i.setClass(context, DaemonService.class);
  11. // 启动service
  12. // 多次调用startService并不会启动多个service 而是会多次调用onStart
  13. context.startService(i);
  14. }
  15. }
  16. }


 

  1. package com.arui.framework.android.daemonservice;
  2. import android.app.Service;
  3. import android.content.Intent;
  4. import android.os.IBinder;
  5. import android.util.Log;
  6. public class DaemonService extends Service {
  7. @Override
  8. public IBinder onBind(Intent intent) {
  9. return null;
  10. }
  11. @Override
  12. public void onCreate() {
  13. super.onCreate();
  14. Log.v("=========", "***** DaemonService *****: onCreate");
  15. }
  16. @Override
  17. public void onStart(Intent intent, int startId) {
  18. Log.v("=========", "***** DaemonService *****: onStart");
  19. // 这里可以做Service该做的事
  20. }
  21. }


 

下面是manifest文件的代码。

 

  1. <receiver
  2. android:name="com.arui.framework.android.daemonservice.BootBroadcast"
  3. android:permission="android.permission.RECEIVE_BOOT_COMPLETED">
  4. <intent-filter>
  5. <action android:name="android.intent.action.BOOT_COMPLETED" />
  6. </intent-filter>
  7. </receiver>
  8. <receiver
  9. android:name="com.arui.framework.android.daemonservice.Alarmreceiver" >
  10. <intent-filter>
  11. <action android:name="arui.alarm.action" />
  12. </intent-filter>
  13. </receiver>
  14. <service
  15. android:name="com.arui.framework.android.daemonservice.DaemonService" >
  16. </service>



 继续讨论这个问题。

如果用户停止整个应用(在管理应用程序中停止应用,或者第三方软件停止整个应用),此时整个进程被杀死,所有的服务自然也被杀死了,timmer,或者alarmManager也就停止了。此时就不会再定期启动服务了。

那么,怎么才能做到,用户或者第三方软件无法停止整个应用呢。我们可以再注册一个系统级别的监听(BroadcastReceiver),来监听系统级别的消息,再次启动timmer,或者alarmManager。这样,即使应用被杀死了,隔一段时间,应用还会自动启动。具体的,就不在这里展开了。

 

---------------------------------------------------------------------------

GL(arui319)

http://blog.csdn.net/arui319

<本文可以转载,但是请保留以上作者信息。谢谢。>

---------------------------------------------------------------------------