Android Service 服務(一)—— Service
一、 Service簡介
Service是android 系統中的四大元件之一(Activity、Service、BroadcastReceiver、ContentProvider),它跟Activity的級別差不多,但不能自己執行只能後臺執行,並且可以和其他元件進行互動。service可以在很多場合的應用中使用,比如播放多媒體的時候使用者啟動了其他Activity這個時候程式要在後臺繼續播放,比如檢測SD卡上檔案的變化,再或者在後臺記錄你地理資訊位置的改變等等,總之服務總是藏在後臺的。
Service的啟動有兩種方式:context.startService() 和 context.bindService()
二、 Service啟動流程
context.startService() 啟動流程:
context.startService() -> onCreate() -> onStart() -> Service running -> context.stopService() -> onDestroy() -> Service stop
如果Service還沒有執行,則android先呼叫onCreate(),然後呼叫onStart();
如果Service已經執行,則只調用onStart(),所以一個Service的onStart方法可能會重複呼叫多次
如果stopService的時候會直接onDestroy,如果是呼叫者自己直接退出而沒有呼叫stopService的話,Service會一直在後臺執行,該Service的呼叫者再啟動起來後可以通過stopService關閉Service。
所以呼叫startService的生命週期為:onCreate --> onStart (可多次呼叫) --> onDestroycontext.bindService()啟動流程:
context.bindService() -> onCreate() -> onBind() -> Service running -> onUnbind() -> onDestroy() -> Service stoponBind()將返回給客戶端一個IBind介面例項,IBind允許客戶端回撥服務的方法,比如得到Service的例項、執行狀態或其他操作。這個時候把呼叫者(Context,例如Activity)會和Service繫結在一起,Context退出了,Srevice就會呼叫onUnbind->onDestroy相應退出。
所以呼叫bindService的生命週期為:onCreate --> onBind(只一次,不可多次繫結) --> onUnbind --> onDestory。
在Service每一次的開啟關閉過程中,只有onStart可被多次呼叫(通過多次startService呼叫),其他onCreate,onBind,onUnbind,onDestory在一個生命週期中只能被呼叫一次。
三、 Service生命週期
Service的生命週期並不像Activity那麼複雜,它只繼承了onCreate()、onStart()、onDestroy()三個方法
當我們第一次啟動Service時,先後呼叫了onCreate()、onStart()這兩個方法;當停止Service時,則執行onDestroy()方法。
這裡需要注意的是,如果Service已經啟動了,當我們再次啟動Service時,不會在執行onCreate()方法,而是直接執行onStart()方法。
它可以通過Service.stopSelf()方法或者Service.stopSelfResult()方法來停止自己,只要呼叫一次stopService()方法便可以停止服務,無論呼叫了多少次的啟動服務方法。
四、 Service示例
下面我做了一個簡單的音樂播放的應用,分別使用startService和bindService來啟動本地的服務。
Activity
public class PlayMusicService extends Activity implements OnClickListener { private Button playBtn; private Button stopBtn; private Button pauseBtn; private Button exitBtn; private Button closeBtn; private Intent intent; @Override public void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.music_service); playBtn = (Button) findViewById(R.id.play); stopBtn = (Button) findViewById(R.id.stop); pauseBtn = (Button) findViewById(R.id.pause); exitBtn = (Button) findViewById(R.id.exit); closeBtn = (Button) findViewById(R.id.close); playBtn.setOnClickListener(this); stopBtn.setOnClickListener(this); pauseBtn.setOnClickListener(this); exitBtn.setOnClickListener(this); closeBtn.setOnClickListener(this); } @Override public void onClick(View v) { int op = -1; intent = new Intent("com.homer.service.musicService"); switch (v.getId()) { case R.id.play: // play music op = 1; break; case R.id.stop: // stop music op = 2; break; case R.id.pause: // pause music op = 3; break; case R.id.close: // close activity this.finish(); break; case R.id.exit: // stopService op = 4; stopService(intent); this.finish(); break; } Bundle bundle = new Bundle(); bundle.putInt("op", op); intent.putExtras(bundle); startService(intent); // startService } @Override public void onDestroy(){ super.onDestroy(); if(intent != null){ stopService(intent); } }}
Service
public class MusicService extends Service { private static final String TAG = "MyService"; private MediaPlayer mediaPlayer; @Override public IBinder onBind(Intent arg0) { return null; } @Override public void onCreate() { Log.v(TAG, "onCreate"); Toast.makeText(this, "show media player", Toast.LENGTH_SHORT).show(); if (mediaPlayer == null) { mediaPlayer = MediaPlayer.create(this, R.raw.tmp); mediaPlayer.setLooping(false); } } @Override public void onDestroy() { Log.v(TAG, "onDestroy"); Toast.makeText(this, "stop media player", Toast.LENGTH_SHORT); if (mediaPlayer != null) { mediaPlayer.stop(); mediaPlayer.release(); } } @Override public void onStart(Intent intent, int startId) { Log.v(TAG, "onStart"); if (intent != null) { Bundle bundle = intent.getExtras(); if (bundle != null) { int op = bundle.getInt("op"); switch (op) { case 1: play(); break; case 2: stop(); break; case 3: pause(); break; } } } } public void play() { if (!mediaPlayer.isPlaying()) { mediaPlayer.start(); } } public void pause() { if (mediaPlayer != null && mediaPlayer.isPlaying()) { mediaPlayer.pause(); } } public void stop() { if (mediaPlayer != null) { mediaPlayer.stop(); try { mediaPlayer.prepare(); // 在呼叫stop後如果需要再次通過start進行播放,需要之前呼叫prepare函式 } catch (IOException ex) { ex.printStackTrace(); } } }}
AndroidManifest.xml
註冊activity
<activity android:name=".service.PlayMusicService" android:label="@string/app_name" />
註冊service
<service android:name=".service.MusicService" android:enabled="true" > <intent-filter> <action android:name="com.homer.service.musicService" /> </intent-filter> </service>
五、 程式碼解析1、Activity中,PlayMusicService中通過重寫OnClickListener 介面onClick()方法實現對播放音樂的控制,把音樂各種操作用數字通過Intent傳遞給service
然後通過構造一個Intent , intent = new Intent("com.homer.service.musicService");
其中,com.homer.service.musicService是 AndroidManifest.xml 對service的定義,即上面“註冊service”
2、Activity中,音樂播放的控制,利用Bundle繫結數字op後,通過 startService(intent); 服務後傳送出去Bundle bundle = new Bundle();bundle.putInt("op", op);intent.putExtras(bundle);startService(intent);
3、 Service中,會處理Activity啟動的 startService(intent);服務,依次呼叫service的啟動過程:onCreate --> onStart(可多次呼叫) --> onDestroy
onCreate(), 建立mediaPlayer
onStart(), 通過獲取Bundle bundle = intent.getExtras();,提取int op = bundle.getInt("op");,然後執行響應的音樂播放操作
onDestroy(),停止並釋放mediaPlayer音樂資源,如果當執行context.stopService()時呼叫此方法
4、Activity中,onClick()函式中close與exit是執行含義是不同的:
close : 只是執行了this.finish(); 關閉了本Activity窗體,service並沒有被關掉,音樂依然會繼續在後臺播放
exit : 先呼叫了stopService(intent); 關閉了service服務,在Service中會呼叫3中的onDestroy()停止並釋放音樂資源,後才執行this.finish(); 關閉了本Activity窗體
六、 拓展知識(程序和宣告週期)
Android作業系統嘗試儘可能長時間的保持應用的程序,但當可用記憶體很低時最終要移走一部分程序。怎樣確定那些程式可以執行,那些要被銷燬,Android讓每一個程序在一個重要級的基礎上執行,重要級低的程序最有可能被淘汰,一共有5級,下面這個列表就是按照重要性排列的:1 一個前臺程序顯示的是使用者此時需要處理和顯示的。下列的條件有任何一個成立,這個程序都被認為是在前臺執行的。 a 與使用者正發生互動的。 b 它控制一個與使用者互動的必須的基本的服務。 c 有一個正在呼叫生命週期的回撥函式的service(如onCreate()、onStar()、onDestroy()) d 它有一個正在執行onReceive()方法的廣播接收物件。只有少數的前臺程序可以在任何給定的時間內執行,銷燬他們是系統萬不得已的、最後的選擇——當記憶體不夠系統繼續執行下去時。通常,在這一點上,裝置已經達到了記憶體分頁狀態,所以殺掉一些前臺程序來保證能夠響應使用者的需求。2 一個可用程序沒有任何前臺元件,但它仍然可以影響到使用者的介面。下面兩種情況發生時,可以稱該程序為可用程序。 它是一個非前臺的activity,但對使用者仍然可用(onPause()方法已經被呼叫)這是可能發生的,例如:前臺的activity是一個允許上一個activity可見的對話方塊,即當前activity半透明,能看到前一個activity的介面,它是一個服務於可用activity的服務。3 一個服務程序是一個通過呼叫startService()方法啟動的服務,並且不屬於前兩種情況。儘管服務程序沒有直接被使用者看到,但他們確實是使用者所關心的,比如後臺播放音樂或網路下載資料。所以系統保證他們的執行,直到不能保證所有的前臺可見程式都正常執行時才會終止他們。4 一個後臺程序就是一個非當前正在執行的activity(activity的onStop()方法已經被呼叫),他們不會對使用者體驗造成直接的影響,當沒有足夠記憶體來執行前臺可見程式時,他們將會被終止。通常,後臺程序會有很多個在執行,所以他們維護一個LRU最近使用程式列表來保證經常執行的activity能最後一個被終止。如果一個activity正確的實現了生命週期的方法,並且儲存它當前狀態,殺死這些程序將不會影響到使用者體驗。5 一個空執行緒沒有執行任何可用應用程式組,保留他們的唯一原因是為了設立一個快取機制,來加快元件啟動的時間。系統經常殺死這些記憶體來平衡系統的整個系統的資源,程序快取和基本核心快取之間的資源。Android把程序裡優先順序最高的activity或服務,作為這個程序的優先順序。例如,一個程序擁有一個服務和一個可見的activity,那麼這個程序將會被定義為可見程序,而不是服務程序。此外,如果別的程序依賴某一個程序的話,那麼被依賴的程序會提高優先順序。一個程序服務於另一個程序,那麼提供服務的程序不會低於獲得服務的程序。例如,如果程序A的一個內容提供商服務於程序B的一個客戶端,或者程序A的一個service被程序B的一個元件繫結,那麼程序A至少擁有和程序B一樣的優先順序,或者更高。因為一個執行服務的程序的優先順序高於執行後臺activity的程序,一個activity會準備一個長時間執行的操作來啟動一個服務,而不是啟動一個執行緒–尤其是這個操作可能會拖垮這個activity。例如後臺播放音樂的同時,通過照相機向伺服器傳送一張照片,啟動一個服務會保證這個操作至少執行在service 程序的優先順序下,無論這個activity發生了什麼,廣播接收者應該作為一個空服務而不是簡單的把耗時的操作單獨放在一個執行緒裡。
參考推薦:
Service (android developer)