Android Activity與Service通信(不同進(jìn)程之間)詳解
在Android中,Activity主要負(fù)責(zé)前臺(tái)頁(yè)面的展示,Service主要負(fù)責(zé)需要長(zhǎng)期運(yùn)行的任務(wù),所以在我們實(shí)際開發(fā)中,就會(huì)常常遇到Activity與Service之間的通信,我們一般在Activity中啟動(dòng)后臺(tái)Service,通過(guò)Intent來(lái)啟動(dòng),Intent中我們可以傳遞數(shù)據(jù)給Service,而當(dāng)我們Service執(zhí)行某些操作之后想要更新UI線程,我們應(yīng)該怎么做呢?接下來(lái)我就介紹三種方式來(lái)實(shí)現(xiàn)Service與Activity之間的通信問(wèn)題
Activity與Service通信的方式有三種:
繼承Binder類
這個(gè)方式只有當(dāng)你的Acitivity和Service處于同一個(gè)Application和進(jìn)程時(shí),才可以用,比如你后臺(tái)有一個(gè)播放背景音樂(lè)的Service,這時(shí)就可以用這種方式來(lái)進(jìn)行通信。
用例子來(lái)說(shuō)明其使用方法:
1. 來(lái)看Service的寫法:
public class LocalService extends Service { // 實(shí)例化自定義的Binder類 private final IBinder mBinder = new LocalBinder(); // 隨機(jī)數(shù)的生成器 private final Random mGenerator = new Random(); /** * 自定義的Binder類,這個(gè)是一個(gè)內(nèi)部類,所以可以知道其外圍類的對(duì)象,通過(guò)這個(gè)類,讓Activity知道其Service的對(duì)象 */ public class LocalBinder extends Binder { LocalService getService() { // 返回Activity所關(guān)聯(lián)的Service對(duì)象,這樣在Activity里,就可調(diào)用Service里的一些公用方法和公用屬性 return LocalService.this; } } @Override public IBinder onBind(Intent intent) { return mBinder; } /** public方法,Activity可以進(jìn)行調(diào)用 */ public int getRandomNumber() { return mGenerator.nextInt(100); } }
在Service里定義一個(gè)內(nèi)部類,Binder的子類,通過(guò)這個(gè)類,把Service的對(duì)象傳給Activity,這樣Activity就可以調(diào)用Service里的公用方法和公用屬性了,但這種方式,一定要在同一個(gè)進(jìn)程和同一個(gè)Application里。
2. 再看相應(yīng)Activity的代碼:
public class BindingActivity extends Activity { LocalService mService; boolean mBound = false; @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } @Override protected void onStart() { super.onStart(); // 綁定Service,綁定后就會(huì)調(diào)用mConnetion里的onServiceConnected方法 Intent intent = new Intent(this, LocalService.class); bindService(intent, mConnection, Context.BIND_AUTO_CREATE); } @Override protected void onStop() { super.onStop(); // 解綁Service,這樣可以節(jié)約內(nèi)存 if (mBound) { unbindService(mConnection); mBound = false; } } /** 用戶點(diǎn)擊button,就讀取Service里的隨機(jī)數(shù) */ public void onButtonClick(View v) { if (mBound) { // 用Service的對(duì)象,去讀取隨機(jī)數(shù) int num = mService.getRandomNumber(); Toast.makeText(this, "number: " + num, Toast.LENGTH_SHORT).show(); } } /** 定交ServiceConnection,用于綁定Service的*/ private ServiceConnection mConnection = new ServiceConnection() { @Override public void onServiceConnected(ComponentName className, IBinder service) { // 已經(jīng)綁定了LocalService,強(qiáng)轉(zhuǎn)IBinder對(duì)象,調(diào)用方法得到LocalService對(duì)象 LocalBinder binder = (LocalBinder) service; mService = binder.getService(); mBound = true; } @Override public void onServiceDisconnected(ComponentName arg0) { mBound = false; } }; }
這里就是通過(guò)IBinder來(lái)得到LocalService對(duì)象,再去調(diào)用其Public方法。
使用Messenger
上面的方法只能在同一個(gè)進(jìn)程里才能用,如果要與另外一個(gè)進(jìn)程的Service進(jìn)行通信,則可以用Messenger。
其實(shí)實(shí)現(xiàn)IPC的方式,還有AIDL,但推薦使用Messenger,有兩點(diǎn)好處:
1. 使用Messenger方式比使用AIDL的方式,實(shí)現(xiàn)起來(lái)要簡(jiǎn)單很多
2. 使用Messenger時(shí),所有從Activity傳過(guò)來(lái)的消息都會(huì)排在一個(gè)隊(duì)列里,不會(huì)同時(shí)請(qǐng)求Service,所以是線程安全的。如果
你的程序就是要多線程去訪問(wèn)Service,就可以用AIDL,不然最好使用Messenger的方式。
不過(guò),其實(shí)Messenger底層用的就是AIDL實(shí)現(xiàn)的,看一下實(shí)現(xiàn)方式,先看Service的代碼:
public class MessengerService extends Service { /** 用于Handler里的消息類型 */ static final int MSG_SAY_HELLO = 1; /** * 在Service處理Activity傳過(guò)來(lái)消息的Handler */ class IncomingHandler extends Handler { @Override public void handleMessage(Message msg) { switch (msg.what) { case MSG_SAY_HELLO: Toast.makeText(getApplicationContext(), "hello!", Toast.LENGTH_SHORT).show(); break; default: super.handleMessage(msg); } } } /** * 這個(gè)Messenger可以關(guān)聯(lián)到Service里的Handler,Activity用這個(gè)對(duì)象發(fā)送Message給Service,Service通過(guò)Handler進(jìn)行處理。 */ final Messenger mMessenger = new Messenger(new IncomingHandler()); /** * 當(dāng)Activity綁定Service的時(shí)候,通過(guò)這個(gè)方法返回一個(gè)IBinder,Activity用這個(gè)IBinder創(chuàng)建出的Messenger,就可以與Service的Handler進(jìn)行通信了 */ @Override public IBinder onBind(Intent intent) { Toast.makeText(getApplicationContext(), "binding", Toast.LENGTH_SHORT).show(); return mMessenger.getBinder(); } }
再看一下Activity的代碼:
public class ActivityMessenger extends Activity { /** 向Service發(fā)送Message的Messenger對(duì)象 */ Messenger mService = null; /** 判斷有沒(méi)有綁定Service */ boolean mBound; private ServiceConnection mConnection = new ServiceConnection() { public void onServiceConnected(ComponentName className, IBinder service) { // Activity已經(jīng)綁定了Service // 通過(guò)參數(shù)service來(lái)創(chuàng)建Messenger對(duì)象,這個(gè)對(duì)象可以向Service發(fā)送Message,與Service進(jìn)行通信 mService = new Messenger(service); mBound = true; } public void onServiceDisconnected(ComponentName className) { mService = null; mBound = false; } }; public void sayHello(View v) { if (!mBound) return; // 向Service發(fā)送一個(gè)Message Message msg = Message.obtain(null, MessengerService.MSG_SAY_HELLO, 0, 0); try { mService.send(msg); } catch (RemoteException e) { e.printStackTrace(); } } @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); setContentView(R.layout.main); } @Override protected void onStart() { super.onStart(); // 綁定Service bindService(new Intent(this, MessengerService.class), mConnection, Context.BIND_AUTO_CREATE); } @Override protected void onStop() { super.onStop(); // 解綁 if (mBound) { unbindService(mConnection); mBound = false; } } }
注意:以上寫的代碼只能實(shí)現(xiàn)從Activity向Service發(fā)送消息,如果想從Service向Activity發(fā)送消息,只要把代碼反過(guò)來(lái)寫就可以了。
使用AIDL
AIDL,Android Interface Definition Language。建立AIDL服務(wù)要比建立普通的服務(wù)復(fù)雜一些,具體步驟如下:
?。?)在Eclipse Android工程的Java包目錄中建立一個(gè)擴(kuò)展名為aidl的文件。該文件的語(yǔ)法類似于Java代碼,但會(huì)稍有不同。詳細(xì)介紹見(jiàn)實(shí)例的內(nèi)容。
?。?)如果aidl文件的內(nèi)容是正確的,ADT會(huì)自動(dòng)生成一個(gè)Java接口文件(*.java)。
?。?)建立一個(gè)服務(wù)類(Service的子類)。
?。?)實(shí)現(xiàn)由aidl文件生成的Java接口。
?。?)在AndroidManifest.xml文件中配置AIDL服務(wù),尤其要注意的是,<action>標(biāo)簽中android:name的屬性值就是客戶端要引用該服務(wù)的ID,也就是Intent類的參數(shù)值。
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
- Android檢測(cè)Activity或者Service是否運(yùn)行的方法
- Android中Service和Activity相互通信示例代碼
- Android中Service與Activity之間通信的幾種方式
- Android Activity 與Service進(jìn)行數(shù)據(jù)交互詳解
- 淺談Android Activity與Service的交互方式
- Android使用Messenger實(shí)現(xiàn)service與activity交互
- Android實(shí)現(xiàn)Activity、Service與Broadcaster三大組件之間互相調(diào)用的方法詳解
- Android實(shí)現(xiàn)從activity中停止Service的方法
- Android中Service實(shí)時(shí)向Activity傳遞數(shù)據(jù)實(shí)例分析
- android使用service和activity獲取屏幕尺寸的方法
- 詳解Android Service與Activity之間通信的幾種方式
相關(guān)文章
Android簡(jiǎn)單實(shí)現(xiàn)一個(gè)顏色漸變的ProgressBar的方法
本篇文章主要介紹了Android簡(jiǎn)單實(shí)現(xiàn)一個(gè)顏色漸變的ProgressBar的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12Flutter 狀態(tài)管理scoped model源碼解讀
這篇文章主要為大家介紹了Flutter 狀態(tài)管理scoped model源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11解決Android studio xml界面無(wú)法預(yù)覽問(wèn)題
這篇文章主要介紹了解決Android studio xml界面無(wú)法預(yù)覽問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2020-03-03Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼
這篇文章主要介紹了Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,一起跟隨小編過(guò)來(lái)看看吧2018-05-05Android自定義View彈性滑動(dòng)Scroller詳解
這篇文章主要為大家詳細(xì)介紹了Android自定義View彈性滑動(dòng)Scroller,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12解決android 顯示內(nèi)容被底部導(dǎo)航欄遮擋的問(wèn)題
今天小編就為大家分享一篇解決android 顯示內(nèi)容被底部導(dǎo)航欄遮擋的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2018-07-07