Android Activity與Service通信(不同進(jìn)程之間)詳解
在Android中,Activity主要負(fù)責(zé)前臺(tái)頁面的展示,Service主要負(fù)責(zé)需要長(zhǎng)期運(yùn)行的任務(wù),所以在我們實(shí)際開發(fā)中,就會(huì)常常遇到Activity與Service之間的通信,我們一般在Activity中啟動(dòng)后臺(tái)Service,通過Intent來啟動(dòng),Intent中我們可以傳遞數(shù)據(jù)給Service,而當(dāng)我們Service執(zhí)行某些操作之后想要更新UI線程,我們應(yīng)該怎么做呢?接下來我就介紹三種方式來實(shí)現(xiàn)Service與Activity之間的通信問題
Activity與Service通信的方式有三種:
繼承Binder類
這個(gè)方式只有當(dāng)你的Acitivity和Service處于同一個(gè)Application和進(jìn)程時(shí),才可以用,比如你后臺(tái)有一個(gè)播放背景音樂的Service,這時(shí)就可以用這種方式來進(jìn)行通信。
用例子來說明其使用方法:
1. 來看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ì)象,通過這個(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的子類,通過這個(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;
}
};
}
這里就是通過IBinder來得到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)起來要簡(jiǎn)單很多
2. 使用Messenger時(shí),所有從Activity傳過來的消息都會(huì)排在一個(gè)隊(duì)列里,不會(huì)同時(shí)請(qǐng)求Service,所以是線程安全的。如果
你的程序就是要多線程去訪問Service,就可以用AIDL,不然最好使用Messenger的方式。
不過,其實(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傳過來消息的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通過Handler進(jìn)行處理。
*/
final Messenger mMessenger = new Messenger(new IncomingHandler());
/**
* 當(dāng)Activity綁定Service的時(shí)候,通過這個(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;
/** 判斷有沒有綁定Service */
boolean mBound;
private ServiceConnection mConnection = new ServiceConnection() {
public void onServiceConnected(ComponentName className, IBinder service) {
// Activity已經(jīng)綁定了Service
// 通過參數(shù)service來創(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ā)送消息,只要把代碼反過來寫就可以了。
使用AIDL
AIDL,Android Interface Definition Language。建立AIDL服務(wù)要比建立普通的服務(wù)復(fù)雜一些,具體步驟如下:
?。?)在Eclipse Android工程的Java包目錄中建立一個(gè)擴(kuò)展名為aidl的文件。該文件的語法類似于Java代碼,但會(huì)稍有不同。詳細(xì)介紹見實(shí)例的內(nèi)容。
?。?)如果aidl文件的內(nèi)容是正確的,ADT會(huì)自動(dòng)生成一個(gè)Java接口文件(*.java)。
?。?)建立一個(gè)服務(wù)類(Service的子類)。
?。?)實(shí)現(xiàn)由aidl文件生成的Java接口。
(5)在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-12
Flutter 狀態(tài)管理scoped model源碼解讀
這篇文章主要為大家介紹了Flutter 狀態(tài)管理scoped model源碼解讀,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11
解決Android studio xml界面無法預(yù)覽問題
這篇文章主要介紹了解決Android studio xml界面無法預(yù)覽問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2020-03-03
Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼
這篇文章主要介紹了Android使用Intent.ACTION_SEND分享圖片和文字內(nèi)容的示例代碼的實(shí)例代碼,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,一起跟隨小編過來看看吧2018-05-05
Android自定義View彈性滑動(dòng)Scroller詳解
這篇文章主要為大家詳細(xì)介紹了Android自定義View彈性滑動(dòng)Scroller,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12
解決android 顯示內(nèi)容被底部導(dǎo)航欄遮擋的問題
今天小編就為大家分享一篇解決android 顯示內(nèi)容被底部導(dǎo)航欄遮擋的問題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-07-07

