1. 程式人生 > >實現安卓widget不被kill的方法

實現安卓widget不被kill的方法

widget本身是不會被kill的,widget原本就是一個broadcastreciver,而且是靜態的,這就意味著隨時發一個訂閱過的廣播widget都能收到,並且會呼叫onReceive()方法。widget之所以不能運行了,往往是應為對應的service被kill。要想widget一直執行就要使service被kill了可以重啟,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掉後自動重寫建立

@Override
 public int onStartCommand(Intent intent, int flags, int startId)
 {
 return START_STICKY_COMPATIBILITY;
 //return super.onStartCommand(intent, flags, startId);
 }

@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.

public void onDestroy()
{
Intent localIntent = new Intent();
localIntent.setClass(this, MyService.class); // 銷燬時重新啟動Service
this.startService(localIntent);
}

3.建立一個廣播

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服務

<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" >

注:解鎖,啟動,切換場景啟用廣播需加許可權,如啟動完成,及手機機狀態等。

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

這種方法確實可以是服務不被,但是會顯得很流氓,當widget從桌面移除了,service還在執行,下面是改進的方法:

1.配置xml:
application節點下配置

<receiver android:name="com.tarena.karen.widdget.TimeWidget" >
            <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" />
                <!-- 服務銷燬的時候傳送 -->
                <action android:name="com.tony.inzone.restart" />
                <action android:name="android.appwidget.action.APPWIDGET_UPDATE" />
            </intent-filter>

            <meta-data
                android:name="android.appwidget.provider"
                android:resource="@xml/widget_info" />
        </receiver>

2.服務被銷燬時候給widget傳送一個廣播

@Override
public void onDestroy() {
        timer.cancel();
        timer = null;
        super.onDestroy();
        Intent localIntent = new Intent();
        localIntent.setAction("com.tony.inzone.restart");
        sendBroadcast(localIntent);
}

3.收到服務被系統回收/系統啟動完成/解鎖完成/監聽情景切換 等情況
widget中再次啟動服務

    /**
     * 訂閱的廣播發來的時候回撥
     */
    @Override
    public void onReceive(Context context, Intent intent) {
        super.onReceive(context, intent);
        Intent intent2=new Intent(context,WidgetService.class);
        context.startService(intent2);
    }

widget被移除時銷燬程式,service不會重啟

    /**
     * 同種型別的最後一個小元件被刪除時回撥
     */
    @Override
    public void onDisabled(Context context) {
        super.onDisabled(context);
        Log.i("TAG", "onDisabled");
        Intent intent=new Intent(context, WidgetService.class);
        context.stopService(intent);
    }