Android教程網
  1. 首頁
  2. Android 技術
  3. Android 手機
  4. Android 系統教程
  5. Android 游戲
 Android教程網 >> Android技術 >> 關於Android編程 >> Android IPC機制綁定Service實現本地通信

Android IPC機制綁定Service實現本地通信

編輯:關於Android編程

**寫作原因:跨進程通信的實現和理解是Android進階中重要的一環。下面博主分享IPC一些相關知識、操作及自己在學習IPC過程中的一些理解。
這一章是為下面的Messenger和AIDL的使用做准備,主要講解Android Service的綁定和Activity與本地Service之間通信相關知識。**

簡介

我們都知道啟動Service有兩種方式:startService()和bindService()。相比第一種方式,bindService()能夠更加靈活地實現與啟動端Activity的數據通信,第一種啟動方式啟動Activity與Service之間並沒有直接關聯,難以直接實現通信(當然了,使用BroadCast或者事件總線也是可以實現的)。而使用綁定的方式啟動Service則可以實現Service之間的通信。 

 下面就講述一下綁定Service實現本地通信的流程。

 

實例

 結合圖片我們來模擬開啟一個後台執行更新功能為例來講解本地Service的綁定。

 先分析一下該需求的場景:首先我們需要點擊Start按鈕開啟後台更新數據,後台將數據返回給Activity並在ProgressBar展示進度;當我們點擊Pause時後台暫停更新;點擊Stop按鈕時關閉後台。本例中也用到Service通過BroadCast向Activity傳遞數據。由於只是實例,對於線程操作一塊有一些bug,希望大家能夠幫忙指正。

Activity的布局如下:

Service 實現:

先看看Service的實現:

思路是這樣的:先繼承Binder類創建MyBinder類,把MyBinder看做Service與Activity通信的代理人,所以在MyBinder內部中寫好方法間接調用Service中的方法以供Activity去調用(如本例中callPauseUpgrade())。關於IBinder對象的獲取除了直接重寫後面還有兩種方式,這裡先不闡述了。在Service中我們開啟一個Thread,在這個Thread中模擬持續更新進度條直到isStop為False或者progress大於等於100時,然後將進度廣播出去,讓Activity接收到廣播進行進度條更新。Service中供Activity調用的方法實現暫停和停止Thread的功能,具體過程參照代碼。

public class UpgradeService extends Service {
 private Thread thread;
 private Intent intent;
 private int progress;
 private Boolean isStop;
 public class MyBinder extends Binder{
  public void callPauseUpgrade(){
   pauseUpgrade();
  }

  public void callStopUpgrade() {
   stopUpgrade();
  }
 }

 private void stopUpgrade() {
  progress = 0;
  isStop = false;
  intent.putExtra("progress",progress);
  sendBroadcast(intent);
 }

 @Override
 public void onCreate() {
  super.onCreate();
  progress = 0;
  isStop = true;
  intent = new Intent();
  thread = new Thread(new Runnable() {
   @Override
   public void run() {
    while(isStop){
     try {
      Thread.sleep(1000);
     } catch (InterruptedException e) {
      e.printStackTrace();
     }
     progress = progress +5;
     intent.putExtra("progress",progress);
     intent.setAction("UPGRADE_ACTION");
     sendBroadcast(intent);
     if(progress>=100) break;
    }

   }
  });
  thread.start();
 }

 private void pauseUpgrade() {
  //TODO:Pause the upgrade
  Toast.makeText(getApplicationContext(),"暫停",Toast.LENGTH_SHORT).show();
  isStop = false;
 }

 @Override
 public IBinder onBind(Intent intent) {
  return new MyBinder();
 }
}

Activity實現

 下面是在客戶端實現與Service的通信代碼。主要思路是:先注冊廣播實現Service數據的返回;單擊Start按鈕時實現與Service的綁定,創建ServiceConnection對象實現ServiceConnection接口,分別回調綁定成功和失敗兩種情況下的邏輯。當成功時獲取MyBinder對象,並將設置的isBound值設為true;當失敗時將isBound設置為false。單擊Pause調用MyBinder對象的callPauseUpgrade()方法間接調用Service中的pauseUpgrade()方法。單擊Stop調用callStopUpgrade()方法並且解除綁定。注意解除綁定事件只能執行一次,否則程序會崩潰。

具體實現並不難,主要是認真理解上面的流程圖即可。

public class MainActivity extends AppCompatActivity implements View.OnClickListener{
 private Button mBtStart;
 private Button mBtStop;
 private Button mBtPause;
 private ProgressBar mPbProgress;
 private UpgradeService.MyBinder myBinder;
 private boolean isBound;
 private UpgradeReceiver upgradeReceiver;
 @Override
 protected void onCreate(Bundle savedInstanceState) {
  super.onCreate(savedInstanceState);
  setContentView(R.layout.activity_main);
  IntentFilter intentFilter = new IntentFilter();
  intentFilter.addAction("UPGRADE_ACTION");
  upgradeReceiver = new UpgradeReceiver();
  registerReceiver(upgradeReceiver,intentFilter);
  initView();
  initEvent();

 }
 public class UpgradeReceiver extends BroadcastReceiver{

  @Override
  public void onReceive(Context context, Intent intent) {
   mPbProgress.setProgress(intent.getIntExtra("progress",0));
  }
 }
 @Override
 protected void onDestroy() {
  super.onDestroy();
  unregisterReceiver(upgradeReceiver);
 }

 private void initEvent() {
  mBtPause.setOnClickListener(this);
  mBtStart.setOnClickListener(this);
  mBtStop.setOnClickListener(this);
 }

 private void initView() {
  mBtPause = (Button) findViewById(R.id.bt_pause);
  mBtStart = (Button) findViewById(R.id.bt_start);
  mBtStop = (Button) findViewById(R.id.bt_stop);
  mPbProgress = (ProgressBar) findViewById(R.id.pb_progress);
 }

 @Override
 public void onClick(View v) {
  switch(v.getId()){
   case R.id.bt_pause:
    if(isBound&&myBinder!=null){
     myBinder.callPauseUpgrade();
    }
    break;
   case R.id.bt_start:
    bindService(new Intent(MainActivity.this,UpgradeService.class),conn,BIND_AUTO_CREATE);
    break;
   case R.id.bt_stop:
    if(isBound) {
     myBinder.callStopUpgrade();
     unbindService(conn);
     isBound = false;
    }
    break;
  }
 }

 private ServiceConnection conn = new ServiceConnection() {
  @Override
  public void onServiceConnected(ComponentName name, IBinder service) {
   if(service != null){
    myBinder = (UpgradeService.MyBinder) service;
    isBound = true;
   }
  }

  @Override
  public void onServiceDisconnected(ComponentName name) {
   isBound = false;
  }
 };
}

總結:

這篇文章是為下面的Messenger和AIDL跨進程通信做准備的,實際上個人感覺真正開發時可以使用EventBus或者RxJava取代這種同進程各個組件通信的問題,有興趣的讀者可以自行嘗試。

  1. 上一頁:
  2. 下一頁:
熱門文章
閱讀排行版
Copyright © Android教程網 All Rights Reserved