Android

个人Android学习总结


service周期性工作

<h3>Android中如何做到Service被关闭后又自动启动</h3> <p>首先要说的是,用户可能把这种做法视为流氓软件。大部分时候,程序员也不想把软件做成流氓软件,没办法,领导说了算。</p> <pre><code> 我们在使用某些Android应用的时候,可能会发现安装了某应用以后,会有一些服务也会随之运行。而且,这些服务每次都会随着手机开机而启动。有的服务做的更绝,当用户在运行的服务中手动停止该服务以后,过了一段时间,服务又自动运行了。虽然,从用户的角度来说,这种方式比较流氓。但是,从程序员的角度来说,这是如何做到的呢?经过研究,我发现有一种方式是可以实现的。下面就和大家分享。 先简单介绍,一会儿会贴上全部代码。 如何做到开机启动? 这个比较简单,网上的资料够多,只要实现一个BroadcastReceiver,监听手机启动完成的事件ACTION_BOOT_COMPLETED即可。需要注意的是,好像不能用模拟器,要用手机测试。 那如何做到启动一个Service,并且在用户关闭后能自动又启动了呢? 一般的,都会在上面说到的BroadcastReceiver的实现里面,监听手机启动完成后,启动一个Service,这是一般的做法。问题是,用户能够在服务里看到这个Service是常驻的。如果用户很敏感,就会停止该Service,甚至直接卸载掉相关的应用。那么,怎样才能定期实现某功能,又不让用户直接看到这个Service呢?聪明的你一定立即就想到了,如果不直接启动Service,而是启动一个timmer,或者alarmManager,然后每隔一段时间去启动Service,做完事情以后关闭掉Service就可以了。 还是看下面的全部代码吧,不过多解释了。这些代码中还是有不少概念的,不熟悉AlarmManager、PendingIntent、BroadcastReceiver、Service等等这些概念的同学可以百度一下。</code></pre> <p>如以下代码:</p> <pre><code>package com.arui.framework.android.daemonservice; import android.app.AlarmManager; import android.app.PendingIntent; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.os.SystemClock; public class BootBroadcast extends BroadcastReceiver { @Override public void onReceive(Context context, Intent mintent) { if (Intent.ACTION_BOOT_COMPLETED.equals(mintent.getAction())) { // 启动完成 Intent intent = new Intent(context, Alarmreceiver.class); intent.setAction(&amp;quot;arui.alarm.action&amp;quot;); PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0); long firstime = SystemClock.elapsedRealtime(); AlarmManager am = (AlarmManager) context .getSystemService(Context.ALARM_SERVICE); // 10秒一个周期,不停的发送广播 am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstime, 10 * 1000, sender); } } }</code></pre> <hr /> <pre><code class="language-java">package com.arui.framework.android.daemonservice; import android.app.AlarmManager; import android.app.PendingIntent; import android.content.BroadcastReceiver; import android.content.Context; import android.content.Intent; import android.os.SystemClock; public class BootBroadcast extends BroadcastReceiver { @Override public void onReceive(Context context, Intent mintent) { if (Intent.ACTION_BOOT_COMPLETED.equals(mintent.getAction())) { // 启动完成 Intent intent = new Intent(context, Alarmreceiver.class); intent.setAction(&amp;quot;arui.alarm.action&amp;quot;); PendingIntent sender = PendingIntent.getBroadcast(context, 0, intent, 0); long firstime = SystemClock.elapsedRealtime(); AlarmManager am = (AlarmManager) context .getSystemService(Context.ALARM_SERVICE); // 10秒一个周期,不停的发送广播 am.setRepeating(AlarmManager.ELAPSED_REALTIME_WAKEUP, firstime, 10 * 1000, sender); } } } </code></pre> <pre><code class="language-java">package com.arui.framework.android.daemonservice; import android.app.Service; import android.content.Intent; import android.os.IBinder; import android.util.Log; public class DaemonService extends Service { @Override public IBinder onBind(Intent intent) { return null; } @Override public void onCreate() { super.onCreate(); Log.v(&amp;quot;=========&amp;quot;, &amp;quot;***** DaemonService *****: onCreate&amp;quot;); } @Override public void onStart(Intent intent, int startId) { Log.v(&amp;quot;=========&amp;quot;, &amp;quot;***** DaemonService *****: onStart&amp;quot;); // 这里可以做Service该做的事 } } </code></pre> <p>下面是manifest文件的代码。</p> <pre><code class="language-java">&amp;lt;receiver android:name=&amp;quot;com.arui.framework.android.daemonservice.BootBroadcast&amp;quot; android:permission=&amp;quot;android.permission.RECEIVE_BOOT_COMPLETED&amp;quot;&amp;gt; &amp;lt;intent-filter&amp;gt; &amp;lt;action android:name=&amp;quot;android.intent.action.BOOT_COMPLETED&amp;quot; /&amp;gt; &amp;lt;/intent-filter&amp;gt; &amp;lt;/receiver&amp;gt; &amp;lt;receiver android:name=&amp;quot;com.arui.framework.android.daemonservice.Alarmreceiver&amp;quot; &amp;gt; &amp;lt;intent-filter&amp;gt; &amp;lt;action android:name=&amp;quot;arui.alarm.action&amp;quot; /&amp;gt; &amp;lt;/intent-filter&amp;gt; &amp;lt;/receiver&amp;gt; &amp;lt;service android:name=&amp;quot;com.arui.framework.android.daemonservice.DaemonService&amp;quot; &amp;gt; &amp;lt;/service&amp;gt; </code></pre> <pre><code> 继续讨论这个问题。 如果用户停止整个应用(在管理应用程序中停止应用,或者第三方软件停止整个应用),此时整个进程被杀死,所有的服务自然也被杀死了,timmer,或者alarmManager也就停止了。此时就不会再定期启动服务了。 那么,怎么才能做到,用户或者第三方软件无法停止整个应用呢。我们可以再注册一个系统级别的监听(BroadcastReceiver),来监听系统级别的消息,再次启动timmer,或者alarmManager。这样,即使应用被杀死了,隔一段时间,应用还会自动启动。具体的,如下。</code></pre> <p>保活广播</p> <pre><code class="language-java">KeepAliveBroadcatReceiver mAliveBroadcatReceiver = new KeepAliveBroadcatReceiver() ; public static final String ACTION_TIME_TICK = &amp;quot;android.intent.action.TIME_TICK&amp;quot;; public static final String ACTION_POWER_DISCONNECTED = &amp;quot;android.intent.action.ACTION_POWER_DISCONNECTED&amp;quot;; public static final String ACTION_BATTERY_CHANGED = &amp;quot;android.intent.action.BATTERY_CHANGED&amp;quot;; public static final String USER_PRESENT = &amp;quot;android.intent.action.USER_PRESENT&amp;quot;; //kaipin public static final String CONNECTIVITY_CHANGE = &amp;quot;android.net.conn.CONNECTIVITY_CHANGE&amp;quot;; //qiehuanwangluo public void registerBroadcastReceiver(Context context, boolean isRegised){ if (isRegised) { IntentFilter filter = new IntentFilter() ; filter.addAction(ACTION_TIME_TICK); filter.addAction(ACTION_POWER_DISCONNECTED); filter.addAction(ACTION_BATTERY_CHANGED); filter.addAction(USER_PRESENT); filter.addAction(ACTION_POWER_DISCONNECTED); context.registerReceiver(mAliveBroadcatReceiver, filter) ; LogUtil.d(TAG, &amp;quot;---------注册mAliveBroadcatReceiver&amp;quot;); }else { context.unregisterReceiver(mAliveBroadcatReceiver); LogUtil.d(TAG, &amp;quot;---------反注册mAliveBroadcatReceiver&amp;quot;); } } //不能再请当中注册,看源码 class KeepAliveBroadcatReceiver extends BroadcastReceiver{ @Override public void onReceive(Context context, Intent intent) { String iAction = intent.getAction() ; //启动app myApp.checkServiceAndStart(context); } }</code></pre>

页面列表

ITEM_HTML