android: 使用 IntentService


9.5.2 使用 IntentService

話說回來,在本章一開始的時候我們就已經知道,服務中的代碼都是默認運行在主線程 當中的,如果直接在服務里去處理一些耗時的邏輯,就很容易出現 ANR(Application Not Responding)的情況。
所以這個時候就需要用到 Android 多線程編程的技術了,我們應該在服務的每個具體的 方法里開啟一個子線程,然后在這里去處理那些耗時的邏輯。因此,一個比較標准的服務就 可以寫成如下形式:

public class MyService extends Service {


@Override
public IBinder onBind(Intent intent) {
return null;
}


@Override
public int onStartCommand(Intent intent, int flags, int startId) {
new Thread(new Runnable() {
@Override
public void run() {
// 處理具體的邏輯
}
}).start();
return super.onStartCommand(intent, flags, startId);
}

}
但是,這種服務一旦啟動之后,就會一直處於運行狀態,必須調用 stopService()或者 stopSelf()方法才能讓服務停止下來。所以,如果想要實現讓一個服務在執行完畢后自動停止 的功能,就可以這樣寫:

public class MyService extends Service {


@Override
public IBinder onBind(Intent intent) {
return null;
}


@Override
public int onStartCommand(Intent intent, int flags, int startId) {
new Thread(new Runnable() {
@Override
public void run() {
// 處理具體的邏輯
stopSelf();
}
}).start();
return super.onStartCommand(intent, flags, startId);
}

}
雖說這種寫法並不復雜,但是總會有一些程序員忘記開啟線程,或者忘記調用 stopSelf() 方法。為了可以簡單地創建一個異步的、會自動停止的服務,Android 專門提供了一個 IntentService 類,這個類就很好地解決了前面所提到的兩種尷尬,下面我們就來看一下它的 用法。
新建一個 MyIntentService 類繼承自 IntentService,代碼如下所示:

public class MyIntentService extends IntentService {


public MyIntentService() {
super("MyIntentService"); // 調用父類的有參構造函數
}


@Override
protected void onHandleIntent(Intent intent) {
// 打印當前線程的id
Log.d("MyIntentService", "Thread id is " + Thread.currentThread(). getId());
}


@Override
public void onDestroy() {
super.onDestroy();
Log.d("MyIntentService", "onDestroy executed");
}

}


這里首先是要提供一個無參的構造函數,並且必須在其內部調用父類的有參構造函數。
然后要在子類中去實現 onHandleIntent()這個抽象方法,在這個方法中可以去處理一些具體的 邏輯,而且不用擔心 ANR 的問題,因為這個方法已經是在子線程中運行的了。這里為了證 實一下,我們在 onHandleIntent()方法中打印了當前線程的 id。另外根據 IntentService 的特性, 這個服務在運行結束后應該是會自動停止的,所以我們又重寫了 onDestroy()方法,在這里也 打印了一行日志,以證實服務是不是停止掉了。
接下來修改 activity_main.xml 中的代碼,加入一個用於啟動 MyIntentService 這個服務的 按鈕,如下所示:

<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android" android:layout_width="match_parent" android:layout_height="match_parent"
android:orientation="vertical" >
……
<Button android:id="@+id/start_intent_service" android:layout_width="match_parent" android:layout_height="wrap_content" android:text="Start IntentService" />
</LinearLayout>
然后修改 MainActivity 中的代碼,如下所示:

public class MainActivity extends Activity implements OnClickListener {
……
private Button startIntentService;


@Override
protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main);
……
startIntentService = (Button) findViewById(R.id.start_intent_service);
startIntentService.setOnClickListener(this);
}


@Override
public void onClick(View v) {
switch (v.getId()) {
……


case R.id.start_intent_service:
// 打印主線程的id
Log.d("MainActivity", "Thread id is " + Thread.currentThread().

getId());


Intent intentService = new Intent(this, MyIntentService.class);
startService(intentService);
break;

default:
break;
}
}
}
可以看到,我們在 Start IntentService 按鈕的點擊事件里面去啟動 MyIntentService 這個服 務,並在這里打印了一下主線程的 id,稍后用於和 IntentService 進行比對。你會發現,其實 IntentService 的用法和普通的服務沒什么兩樣。
最后仍然不要忘記,服務都是需要在 AndroidManifest.xml 里注冊的,如下所示:

<manifest xmlns:android="http://schemas.android.com/apk/res/android" package="com.example.servicetest"
android:versionCode="1"
android:versionName="1.0" >
……
<application android:allowBackup="true" android:icon="@drawable/ic_launcher" android:label="@string/app_name" android:theme="@style/AppTheme" >
……
<service android:name=".MyIntentService"></service>
</application>
</manifest>
現在重新運行一下程序,界面如圖 9.13 所示。

圖 9.13

點擊 Start IntentService 按鈕后,觀察 LogCat 中的打印日志,如圖 9.14 所示。

 

圖 9.14

可以看到,不僅 MyIntentService 和 MainActivity 所在的線程 id 不一樣,而且 onDestroy() 方法也得到了執行,說明 MyIntentService 在運行完畢后確實自動停止了。集開啟線程和自動 停止於一身,IntentService 還是博得了不少程序員的喜愛。


免責聲明!

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



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