Activity與Service通信

繼承Binder類

  這個方式只有當你的Acitivity和Service處於同一個Application和進程時,纔可以用,比如你後臺有一個播放背景音樂的Service,這時就可以用這種方式來進行通信。

用例子來說明其使用方法:

  1. 來看Service的寫法:

 

  1. public class LocalService extends Service {  
  2.     // 實例化自定義的Binder類  
  3.     private final IBinder mBinder = new LocalBinder();  
  4.     // 隨機數的生成器  
  5.     private final Random mGenerator = new Random();  
  6.   
  7.     /** 
  8.      * 自定義的Binder類,這個是一個內部類,所以可以知道其外圍類的對象,通過這個類,讓Activity知道其Service的對象 
  9.      */  
  10.     public class LocalBinder extends Binder {  
  11.         LocalService getService() {  
  12.             // 返回Activity所關聯的Service對象,這樣在Activity裏,就可調用Service裏的一些公用方法和公用屬性  
  13.             return LocalService.this;  
  14.         }  
  15.     }  
  16.   
  17.     @Override  
  18.     public IBinder onBind(Intent intent) {  
  19.         return mBinder;  
  20.     }  
  21.   
  22.     /** public方法,Activity可以進行調用 */  
  23.     public int getRandomNumber() {  
  24.       return mGenerator.nextInt(100);  
  25.     }  
  26. }  

 

   在Service裏定義一個內部類,Binder的子類,通過這個類,把Service的對象傳給Activity,這樣Activity就可以調用Service裏的公用方法和公用屬性了,但這種方式,一定要在同一個進程和同一個Application裏。

   2. 再看相應Activity的代碼:

 

  1. public class BindingActivity extends Activity {  
  2.     LocalService mService;  
  3.     boolean mBound = false;  
  4.   
  5.     @Override  
  6.     protected void onCreate(Bundle savedInstanceState) {  
  7.         super.onCreate(savedInstanceState);  
  8.         setContentView(R.layout.main);  
  9.     }  
  10.   
  11.     @Override  
  12.     protected void onStart() {  
  13.         super.onStart();  
  14.         // 綁定Service,綁定後就會調用mConnetion裏的onServiceConnected方法  
  15.         Intent intent = new Intent(this, LocalService.class);  
  16.         bindService(intent, mConnection, Context.BIND_AUTO_CREATE);  
  17.     }  
  18.   
  19.     @Override  
  20.     protected void onStop() {  
  21.         super.onStop();  
  22.         // 解綁Service,這樣可以節約內存  
  23.         if (mBound) {  
  24.             unbindService(mConnection);  
  25.             mBound = false;  
  26.         }  
  27.     }  
  28.   
  29.     /** 用戶點擊button,就讀取Service裏的隨機數 */  
  30.     public void onButtonClick(View v) {  
  31.         if (mBound) {  
  32.             // 用Service的對象,去讀取隨機數  
  33.             int num = mService.getRandomNumber();  
  34.             Toast.makeText(this"number: " + num, Toast.LENGTH_SHORT).show();  
  35.         }  
  36.     }  
  37.   
  38.     /** 定交ServiceConnection,用於綁定Service的*/  
  39.     private ServiceConnection mConnection = new ServiceConnection() {  
  40.   
  41.         @Override  
  42.         public void onServiceConnected(ComponentName className,  
  43.                 IBinder service) {  
  44.             // 已經綁定了LocalService,強轉IBinder對象,調用方法得到LocalService對象  
  45.             LocalBinder binder = (LocalBinder) service;  
  46.             mService = binder.getService();  
  47.             mBound = true;  
  48.         }  
  49.   
  50.         @Override  
  51.         public void onServiceDisconnected(ComponentName arg0) {  
  52.             mBound = false;  
  53.         }  
  54.     };  
  55. }  

 

   這裏就是通過IBinder來得到LocalService對象,再去調用其Public方法。

使用Messenger

   上面的方法只能在同一個進程裏才能用,如果要與另外一個進程的Service進行通信,則可以用Messenger。

    其實實現IPC的方式,還有AIDL,但推薦使用Messenger,有兩點好處:

      1. 使用Messenger方式比使用AIDL的方式,實現起來要簡單很多

      2. 使用Messenger時,所有從Activity傳過來的消息都會排在一個隊列裏,不會同時請求Service,所以是線程安全的。如果你的程序就是要多線程去訪問Service,就可以用AIDL,不然最好使用Messenger的方式。

  不過,其實Messenger底層用的就是AIDL實現的,看一下實現方式,先看Service的代碼:

 

  1. public class MessengerService extends Service {  
  2.     /** 用於Handler裏的消息類型 */  
  3.     static final int MSG_SAY_HELLO = 1;  
  4.   
  5.     /** 
  6.      * 在Service處理Activity傳過來消息的Handler 
  7.      */  
  8.     class IncomingHandler extends Handler {  
  9.         @Override  
  10.         public void handleMessage(Message msg) {  
  11.             switch (msg.what) {  
  12.                 case MSG_SAY_HELLO:  
  13.                     Toast.makeText(getApplicationContext(), "hello!", Toast.LENGTH_SHORT).show();  
  14.                     break;  
  15.                 default:  
  16.                     super.handleMessage(msg);  
  17.             }  
  18.         }  
  19.     }  
  20.   
  21.     /** 
  22.      * 這個Messenger可以關聯到Service裏的Handler,Activity用這個對象發送Message給Service,Service通過Handler進行處理。 
  23.      */  
  24.     final Messenger mMessenger = new Messenger(new IncomingHandler());  
  25.   
  26.     /** 
  27.      * 當Activity綁定Service的時候,通過這個方法返回一個IBinder,Activity用這個IBinder創建出的Messenger,就可以與Service的Handler進行通信了 
  28.      */  
  29.     @Override  
  30.     public IBinder onBind(Intent intent) {  
  31.         Toast.makeText(getApplicationContext(), "binding", Toast.LENGTH_SHORT).show();  
  32.         return mMessenger.getBinder();  
  33.     }  
  34. }  

 

 再看一下Activity的代碼:

  1. public class ActivityMessenger extends Activity {  
  2.     /** 向Service發送Message的Messenger對象 */  
  3.     Messenger mService = null;  
  4.   
  5.     /** 判斷有沒有綁定Service */  
  6.     boolean mBound;  
  7.   
  8.     private ServiceConnection mConnection = new ServiceConnection() {  
  9.         public void onServiceConnected(ComponentName className, IBinder service) {  
  10.             // Activity已經綁定了Service  
  11.             // 通過參數service來創建Messenger對象,這個對象可以向Service發送Message,與Service進行通信  
  12.             mService = new Messenger(service);  
  13.             mBound = true;  
  14.         }  
  15.   
  16.         public void onServiceDisconnected(ComponentName className) {  
  17.             mService = null;  
  18.             mBound = false;  
  19.         }  
  20.     };  
  21.   
  22.     public void sayHello(View v) {  
  23.         if (!mBound) return;  
  24.         // 向Service發送一個Message  
  25.         Message msg = Message.obtain(null, MessengerService.MSG_SAY_HELLO, 00);  
  26.         try {  
  27.             mService.send(msg);  
  28.         } catch (RemoteException e) {  
  29.             e.printStackTrace();  
  30.         }  
  31.     }  
  32.   
  33.     @Override  
  34.     protected void onCreate(Bundle savedInstanceState) {  
  35.         super.onCreate(savedInstanceState);  
  36.         setContentView(R.layout.main);  
  37.     }  
  38.   
  39.     @Override  
  40.     protected void onStart() {  
  41.         super.onStart();  
  42.         // 綁定Service  
  43.         bindService(new Intent(this, MessengerService.class), mConnection,  
  44.             Context.BIND_AUTO_CREATE);  
  45.     }  
  46.   
  47.     @Override  
  48.     protected void onStop() {  
  49.         super.onStop();  
  50.         // 解綁  
  51.         if (mBound) {  
  52.             unbindService(mConnection);  
  53.             mBound = false;  
  54.         }  
  55.     }  
  56. }  

 注意:以上寫的代碼只能實現從Activity向Service發送消息,如果想從Service向Activity發送消息,只要把代碼反過來寫就可以了。

 

使用AIDL

  這個方法略,如果知道上面兩種方法,這個方法基本很少會用到。

發佈了59 篇原創文章 · 獲贊 15 · 訪問量 38萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章