【轉】Android Service被關閉后自動重啟,解決被異常kill 服務


 

http://www.kaifajie.cn/android/10182-2.html

 

每次調用startService(Intent)的時候,都會調用該Service對象的onStartCommand(Intent,int,int)方法,這個方法return 一個int值,return 的值有四種:
  
  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后一定能重啟。
  
  1.在service中重寫下面的方法,這個方法有三個返回值, START_STICKY是service被kill掉后自動重寫創建
  
  @Override
  
  public int onStartCommand(Intent intent, int flags, int startId) {
  
  return START_STICKY;
  
  }----------------
  
  @Override
  
  public int onStartCommand(Intent intent, int flags, int startId) {
  
  // TODO Auto-generated method stub
  
  Log.v("TrafficService","startCommand");
  
  flags = START_STICKY;
  
  return super.onStartCommand(intent, flags, startId);
  
  // return START_REDELIVER_INTENT;
  
  }
  
  2.在Service的onDestroy()中重啟Service.
  
  public void onDestroy() {
  
  Intent localIntent = new Intent();
  
  localIntent.setClass(this, MyService.class); //銷毀時重新啟動Service
  
  this.startService(localIntent);
  
  }---------------------------------------------
  
  用qq管家殺掉進程的時候,調用的是系統自帶的強制kill功能(即settings里的),在kill時,會將應用的整個進程停掉,當然包括service在內,如果在running里將service強制kill掉,顯示進程還在。不管是kill整個進程還是只kill掉進應用的 service,都不會重新啟動service。不知道你是怎么怎么實現重啟的,實在是不解。。。
  
  ps:在eclipse中,用stop按鈕kill掉進程的時候,倒是會重啟service
  
  KILL問題:
  
  1. settings 中stop service
  
  onDestroy方法中,調用startService進行Service的重啟。
  
  2.settings中force stop 應用
  
  捕捉系統進行廣播(action為android.intent.action.PACKAGE_RESTARTED)
  
  3. 借助第三方應用kill掉running task
  
  提升service的優先級
  
  --------------------------------------------
  
  service開機啟動
  
  http://blog.csdn.net/flying_vip_521/article/details/7053355
  
  今天我們主要來探討android怎么讓一個service開機自動啟動功能的實現。Android手機在啟動的過程中會觸發一個Standard Broadcast Action,名字叫android.intent.action.BOOT_COMPLETED(記得只會觸發一次呀),在這里我們可以通過構建一個廣播接收者來接收這個這個action.下面我就來簡單寫以下實現的步驟:
  
  第一步:首先創建一個廣播接收者,重構其抽象方法 onReceive(Context context, Intent intent),在其中啟動你想要啟動的Service或app。
  
  import android.content.BroadcastReceiver;
  
  import android.content.Context;
  
  import android.content.Intent;
  
  import android.util.Log;
  
  public class BootBroadcastReceiver extends BroadcastReceiver {
  
  //重寫onReceive方法
  
  @Override
  
  public void onReceive(Context context, Intent intent) {
  
  //后邊的XXX.class就是要啟動的服務
  
  Intent service = new Intent(context,XXXclass);
  
  context.startService(service);
  
  Log.v("TAG", "開機自動服務自動啟動.....");
  
  //啟動應用,參數為需要自動啟動的應用的包名
  
  Intent intent = getPackageManager().getLaunchIntentForPackage(packageName);
  
  context.startActivity(intent );
  
  }
  
  }
  
  第二步:配置xml文件,在receiver接收這種添加intent-filter配置
  
  <receiver android:name="BootBroadcastReceiver">
  
  <intent-filter>
  
  <action android:name="android.intent.action.BOOT_COMPLETED"></action>
  
  <category android:name="android.intent.category.LAUNCHER" />
  
  </intent-filter>
  
  </receiver>
  
  第三步:添加權限 <uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
  
  ---------------------------------------------------
  
  如何實現一個不會被殺死的進程
  
  看Android的文檔知道,當進程長期不活動,或系統需要資源時,會自動清理門戶,殺死一些Service,和不可見的Activity等所在的進程。
  
  但是如果某個進程不想被殺死(如數據緩存進程,或狀態監控進程,或遠程服務進程),應該怎么做,才能使進程不被殺死。
  
  add android:persistent="true" into the <application> section in your AndroidManifest.xml
  
  切記,這個不可濫用,系統中用這個的service,app一多,整個系統就完蛋了。
  
  目前系統中有phone等非常有限的,必須一直活着的應用在試用。
  
  ------------------------------------------------
  
  提升service優先級的方法
  
  Android 系統對於內存管理有自己的一套方法,為了保障系統有序穩定的運信,系統內部會自動分配,控制程序的內存使用。當系統覺得當前的資源非常有限的時候,為了保 證一些優先級高的程序能運行,就會殺掉一些他認為不重要的程序或者服務來釋放內存。這樣就能保證真正對用戶有用的程序仍然再運行。如果你的 Service 碰上了這種情況,多半會先被殺掉。但如果你增加 Service 的優先級就能讓他多留一會,我們可以用 setForeground(true) 來設置 Service 的優先級。
  
  為什么是 foreground ? 默認啟動的 Service 是被標記為 background,當前運行的 Activity 一般被標記為 foreground,也就是說你給 Service 設置了 foreground 那么他就和正在運行的 Activity 類似優先級得到了一定的提高。當讓這並不能保證你得 Service 永遠不被殺掉,只是提高了他的優先級。
  
  從Android 1.5開始,一個已啟動的service可以調用startForeground(int, Notification)將service置為foreground狀態,調用stopForeground(boolean)將service置為 background狀態。
  
  我們會在調用startForeground(int, Notification)傳入參數notification,它會在狀態欄里顯示正在進行的foreground service。background service不會在狀態欄里顯示。
  
  在Android 1.0中,將一個service置為foreground狀態:
  
  setForeground(true);
  
  mNM.notify(id, notification);
  
  將一個service置為background狀態:
  
  mNM.cancel(id);
  
  setForeground(false);
  
  對比看出,在1.0 API中調用setForeground(boolean)只是簡單的改變service的狀態,用戶不會有任何覺察。新API中強制將 notification和改變service狀態的動作綁定起來,foreground service會在狀態欄顯示,而background service不會。
  
  Remote service controller & binding
  
  跨進程調用Service。暫時不研究。
  
  -------------------------------------------------------
  
  如何防止Android應用中的Service被系統回收? 很多朋友都在問,如何防止Android應用中的Service被系統回收?下面簡單解答一下。
  
  對於Service被系統回收,一般做法是通過提高優先級可以解決,在AndroidManifest.xml文件中對於intent-filter可以通過android:priority = "1000"這個屬性設置最高優先級,1000是最高值,如果數字越小則優先級越低,同時實用於廣播,推薦大家如果你的應用很重要,可以考慮通過系統常用intent action來觸發。

 

 


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM