9.5.2 使用 IntentServiceandroid
話說回來,在本章一開始的時候咱們就已經知道,服務中的代碼都是默認運行在主線程 當中的,若是直接在服務裏去處理一些耗時的邏輯,就很容易出現 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);
}app
}
可是,這種服務一旦啓動以後,就會一直處於運行狀態,必須調用 stopService()或者 stopSelf()方法才能讓服務中止下來。因此,若是想要實現讓一個服務在執行完畢後自動中止 的功能,就能夠這樣寫:異步
public class MyService extends Service {ide
@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);
}this
}
雖然說這種寫法並不複雜,可是總會有一些程序員忘記開啓線程,或者忘記調用 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 仍是博得了很多程序員的喜好。