Android中IntentService的原理及使用

lostinai發表於2014-06-26

在Android開發中,我們或許會碰到這麼一種業務需求,一項任務分成幾個子任務,子任務按順序先後執行,子任務全部執行完後,這項任務才算成功。那麼,利用幾個子執行緒順序執行是可以達到這個目的的,但是每個執行緒必須去手動控制,而且得在一個子執行緒執行完後,再開啟另一個子執行緒。或者,全部放到一個執行緒中讓其順序執行。這樣都可以做到,但是,如果這是一個後臺任務,就得放到Service裡面,由於Service和Activity是同級的,所以,要執行耗時任務,就得在Service裡面開子執行緒來執行。那麼,有沒有一種簡單的方法來處理這個過程呢,答案就是IntentService。


什麼是IntentService,首先看看官方的解釋:

IntentService is a base class forServices that handle asynchronous requests (expressed asIntents) on demand. Clients send requests throughstartService(Intent) calls; the service is started as needed, handles each Intent in turn using a worker thread, and stops itself when it runs out of work



簡單說,IntentService是繼承於Service並處理非同步請求的一個類,在IntentService內有一個工作執行緒來處理耗時操作,啟動IntentService的方式和啟動傳統Service一樣,同時,當任務執行完後,IntentService會自動停止,而不需要我們去手動控制。另外,可以啟動IntentService多次,而每一個耗時操作會以工作佇列的方式在IntentService的onHandleIntent回撥方法中執行,並且,每次只會執行一個工作執行緒,執行完第一個再執行第二個,以此類推。

還有一個說明是:

All requests are handled on a single worker thread -- they may take as long as necessary (and will not block the application's main loop), but only one request will be processed at a time.

大致意思是:所有請求都在一個單執行緒中,不會阻塞應用程式的主執行緒(UI Thread),同一時間只處理一個請求。

那麼,用IntentService有什麼好處呢?首先,我們省去了在Service中手動開執行緒的麻煩,第二,當操作完成時,我們不用手動停止Service,第三,it's so easy to use!



ok,接下來讓我們來看看如何使用,我寫了一個Demo來模擬兩個耗時操作,Operation1與Operation2,先執行1,2必須等1執行完才能執行:

新建工程,新建一個繼承IntentService的類,我這裡是IntentServiceDemo.java

[java] view plaincopy
  1. public class IntentServiceDemo extends IntentService {  
  2.   
  3.     public IntentServiceDemo() {  
  4.         //必須實現父類的構造方法  
  5.         super("IntentServiceDemo");  
  6.     }  
  7.       
  8.     @Override  
  9.     public IBinder onBind(Intent intent) {  
  10.         System.out.println("onBind");  
  11.         return super.onBind(intent);  
  12.     }  
  13.   
  14.   
  15.     @Override  
  16.     public void onCreate() {  
  17.         System.out.println("onCreate");  
  18.         super.onCreate();  
  19.     }  
  20.   
  21.     @Override  
  22.     public void onStart(Intent intent, int startId) {  
  23.         System.out.println("onStart");  
  24.         super.onStart(intent, startId);  
  25.     }  
  26.   
  27.   
  28.     @Override  
  29.     public int onStartCommand(Intent intent, int flags, int startId) {  
  30.         System.out.println("onStartCommand");  
  31.         return super.onStartCommand(intent, flags, startId);  
  32.     }  
  33.   
  34.   
  35.     @Override  
  36.     public void setIntentRedelivery(boolean enabled) {  
  37.         super.setIntentRedelivery(enabled);  
  38.         System.out.println("setIntentRedelivery");  
  39.     }  
  40.   
  41.     @Override  
  42.     protected void onHandleIntent(Intent intent) {  
  43.         //Intent是從Activity發過來的,攜帶識別引數,根據引數不同執行不同的任務  
  44.         String action = intent.getExtras().getString("param");  
  45.         if (action.equals("oper1")) {  
  46.             System.out.println("Operation1");  
  47.         }else if (action.equals("oper2")) {  
  48.             System.out.println("Operation2");  
  49.         }  
  50.           
  51.         try {  
  52.             Thread.sleep(2000);  
  53.         } catch (InterruptedException e) {  
  54.             e.printStackTrace();  
  55.         }  
  56.     }  
  57.   
  58.     @Override  
  59.     public void onDestroy() {  
  60.         System.out.println("onDestroy");  
  61.         super.onDestroy();  
  62.     }  
  63.   
  64. }  

我把生命週期方法全列印出來了,待會我們來看看它執行的過程是怎樣的。接下來是Activity,在Activity中來啟動IntentService:

[java] view plaincopy
  1. public class TestActivity extends Activity {  
  2.     /** Called when the activity is first created. */  
  3.     @Override  
  4.     public void onCreate(Bundle savedInstanceState) {  
  5.         super.onCreate(savedInstanceState);  
  6.         setContentView(R.layout.main);  
  7.           
  8.         //可以啟動多次,每啟動一次,就會新建一個work thread,但IntentService的例項始終只有一個  
  9.         //Operation 1  
  10.         Intent startServiceIntent = new Intent("com.test.intentservice");  
  11.         Bundle bundle = new Bundle();  
  12.         bundle.putString("param""oper1");  
  13.         startServiceIntent.putExtras(bundle);  
  14.         startService(startServiceIntent);  
  15.           
  16.         //Operation 2  
  17.         Intent startServiceIntent2 = new Intent("com.test.intentservice");  
  18.         Bundle bundle2 = new Bundle();  
  19.         bundle2.putString("param""oper2");  
  20.         startServiceIntent2.putExtras(bundle2);  
  21.         startService(startServiceIntent2);  
  22.     }  
  23. }  

最後,別忘了配置Service,因為它繼承於Service,所以,它還是一個Service,一定要配置,否則是不起作用的,開始我就是忘了,結果半天沒反應。

[html] view plaincopy
  1. <service android:name=".IntentServiceDemo">  
  2.             <intent-filter >  
  3.                 <action android:name="com.test.intentservice"/>  
  4.             </intent-filter>  
  5.         </service>  

ok,最後來看看執行結果:


從結果可以看到,onCreate方法只執行了一次,而onStartCommand和onStart方法執行了兩次,開啟了兩個Work Thread,這就證實了之前所說的,啟動多次,但IntentService的例項只有一個,這跟傳統的Service是一樣的。Operation1也是先於Operation2列印,並且我讓兩個操作間停頓了2s,最後是onDestroy銷燬了IntentService。


這就是IntentService,一個方便我們處理業務流程的類,它是一個Service,但是比Service更智慧。希望本文對大家有用,有任何建議或想法歡迎留言和我交流。

Demo原始碼下載


相關文章