Android的最重要的組件式service和activity,那麼在使用的過程中,我們最常遇到的問題是他們之間的通訊問題。當然今天我們不研究底層的實現問題,我從上層的應用的如何使用的角度進行研究。
首先Activity調用Service
這個是比較基礎的,它有兩種常見的方法;
第一, 通過Intent,這個比較簡單了,可以指定package name和class name的方式來調用,Intent.setClassName這個成員即可。通過putString來裝載資料,startService(intent)即可例子如下:
Intent regIntent = new Intent(“com.service”);
regIntent.putExtra(“data”, "helloData");
startService(regIntent);
第二, 通過IPC,這個比較麻煩,一般用不著,這裡不談
但是,反過來,Service如何將一些狀態告訴Activity呢?方法有兩種
第 一、 service 通過廣播的形式發送broadcast,我們寫一個broadcastReceiver即可,通常的情況,將broadcastReceiver寫成 Activity的內部類,這個onReceiver可以直接調用activity的方法來更新介面。但是內部類只能採用代碼註冊的方法
registerReceiver(),不能再AndroidManifest.xml檔案中進行靜態聲明,因為內部類要依賴於外部類而存在的。如果你 一定要用AndroidManifest來註冊receiver,那麼只能把broadcastReceiver寫成單獨的檔案的public類。這時 候,你想更新介面就比較麻煩了,你只能自己把你要更新的activity運行起來,然後再向這個activity的內部類發廣播的訊息來更新介面
第 二、 service直接向activity發intent。在service裡面進行startActivity是屬於在Activity外 startActivity即在task外啟動activity,因此,必須在intent加入一個參數如下:
Intent intentSend = new Intent(Constants.ACTION_STATUS);
intent1.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK);
intent1.putExtra(“statues“,“end“);
context.startActivity(intent1);
但 是此時會引發的一個問題是,多次startActivity會導致很多的activity實現被運行,這肯定不是我們要的,我只要一個Activity就 可,此時,我們要在androidManifest裡面對這個activity的launchMode設定為singleInstance
<activity android:name="com.demo.Activity"
android:label="@string/online" android:launchMode="singleInstance">
記住啦,有人設定為singleTask,也可以,但他們有一點區別。
同時記住要更新intent,這樣getInstent才可以得到每次的新執行個體
@Override
protected void onNewIntent (Intent intent){
setIntent(intent);
}
在後台啟動並執行Service,有時候也需要通知前台的Activity,向Activity發送一些資料。例如,當資料下載已經完成或者音樂播放結 束時,為了降低程式的耦合,使用Intent在Service和Activity之間通訊是一個不錯的方案。在Activity中註冊一個 BroadcastReceiver,當Service有資料發送給Activity時,構建一個Intent並調用sendBroadcast()方法 將資料發送給Activity。
在chapter8_1的基礎上做適當修改,使得音樂播放結束後,MainActivity可以收到來自MusicService的通知。首先,為 MediaPlayer註冊一個OnCompletionListener,當音樂播放結束後,向MainActivity發送Intent。
- MediaPlayer.OnCompletionListener listener = new MediaPlayer.OnCompletionListener() {
-
- public void onCompletion(MediaPlayer arg0) {
- //MusicService使用廣播方式向MainActivity發送資料
- Intent intent = new Intent(MUSIC_COMPLETED);
- intent.putExtra("msg", getText(R.string.music_completed));
- sendBroadcast(intent);
- }
- };
接下來,要為MainActivity註冊一個BroadcastReceiver,來監聽來自MusicService的資訊。當 MusicReceiver接收到來自MusicService的廣播後,彈出一個Toast提示使用者。MainActivity增加的代碼如下所示:
- class MusicReceiver extends BroadcastReceiver {
- @Override
- public void onReceive(Context arg0, Intent arg1) {
- String action = arg1.getAction();
- if (action.equals(MusicService.MUSIC_COMPLETED)) {
- //從MusicService接收廣播訊息,彈出Toast
- String msg = arg1.getStringExtra("msg");
- Toast.makeText(MainActivity.this, msg, Toast.LENGTH_SHORT)
- .show();
- }
- }
- }
- @Override
- protected void onPause() {
- //登出BroadcastReceiver
- unregisterReceiver(receiver);
- super.onPause();
- }
-
- @Override
- protected void onResume() {
- IntentFilter filter = new IntentFilter(MusicService.MUSIC_COMPLETED);
- if (receiver == null)
- receiver = new MusicReceiver();
- //註冊BroadcastReceiver
- registerReceiver(receiver, filter);
- super.onResume();
- }
如 果有多個Activity需要註冊BroadcastReceiver來監聽來自Service的訊息該怎麼辦?一個一個註冊,顯然比較麻煩。可以通過定 義一個父Activity來註冊BroadcastReceiver,其他的Activity繼承這個父類,達到簡化程式的目的。