进程间通信 binder机制

android进程间通信采用的是binder机制.

应用场景:1.ActivityManagerService(extends Binder)、WindowManagerService 在system_server进程

                         应用程序要跨进程方法它们的方法。

                      2.在应用程序之间要互相访问,比如音乐中的MediaPlaybackService(extends Service),锁屏apk

                         调用其MediaPlaybackService中的方法来播放music。

工具:AIDL可以把一个 定义了包含MediaPlaybackService所有方法的.aidl接口文件编译成一个进程间通讯的类,

                          IMediaPlaybackService.Stub(extends Binder),该类包含了

                一个内部类Proxy,Proxy工作在客户端,把请求transact到服务器端Stub的onTransact方法进行处理。

                    是不是有点神奇,一个类的横跨两个进程,即客户端发送请求和服务器处理请求都在这个类里面处理了。

private android.os.IBinder mRemote;

public void openFile(java.lang.String path) throws android.os.RemoteException
{
android.os.Parcel _data = android.os.Parcel.obtain();
android.os.Parcel _reply = android.os.Parcel.obtain();
try {
_data.writeInterfaceToken(DESCRIPTOR);
_data.writeString(path);
mRemote.transact(Stub.TRANSACTION_openFile, _data, _reply, 0);
_reply.readException();
}
@Override public boolean onTransact(int code, android.os.Parcel data, android.os.Parcel reply, int flags) throws android.os.RemoteException
{
case TRANSACTION_openFile:
{
data.enforceInterface(DESCRIPTOR);
java.lang.String _arg0;
_arg0 = data.readString();
this.openFile(_arg0);
reply.writeNoException();
return true;
}


比如音乐应用中有一个MediaPlaybackService extends Service,如果锁屏要访问这service中的方法,需要如下方法。

1.androidManifest.xml中service添加export = true;

2.增加IMediaPlaybackService.aidl并且修改Android.mk 编译 生成IMediaPlaybackService.java

3.MediaPlaybackService增加mBinder 复写onBind方法

 private final IBinder mBinder = new ServiceStub(this);

 @Override
    public IBinder onBind(Intent intent) {
        mDelayedStopHandler.removeCallbacksAndMessages(null);
        mServiceInUse = true;
        return mBinder;
    }

4.

    static class ServiceStub extends IMediaPlaybackService.Stub {
        WeakReference<MediaPlaybackService> mService;

        ServiceStub(MediaPlaybackService service) {
            mService = new WeakReference<MediaPlaybackService>(service);
        }

        @Override
        public boolean open(long [] list, int position) {
            return mService.get().open(list, position);
        }

        @Override
        public boolean openList(long [] list, int position, boolean forceShuffle, String channelName) {
            return mService.get().openList(list, position, forceShuffle, channelName);
        }

5.通过bindService 和 onServiceConnected 在另外一个进程获得这个service的binder接口 并且调用 open等方法。

  调用进程也需要IMediaPlaybackService.aidl 。

myService = IMediaPlaybackService.Stub.asInterface(service); 获得IMediaPlaybackService.Stub.Proxy对象

之后就是IMediaPlaybackService.Stub.ProxyIMediaPlaybackService.Stub直接通过onTransact Transact 进行binder通信
=============================================================================================


Activity与Service[四大组件之一]通信的方式有三种:

1. bindService + 继承Binder类  同一个进程

2. bindService+AIDL   可以在不同进程,方法类似App 调用 SystemServer进程的PowerManagerService中的方法

3.Messenger  

方法一:

  这个方式只有当你的Acitivity和Service处于同一个Application和进程时,才可以用,比如你后台有一个播放背景音乐的Service,这时就可以用这种方式来进行通信。

用例子来说明其使用方法:

  1. 来看Service的写法:

 

[java]  view plain copy
  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的代码:

 

[java]  view plain copy
  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;  
[java]  view plain copy
  1. <span style="color:#008000;">//</span><span style="color:#008000;">得到了服务类对象就可以使用服务对象提供的方法了。但不可以在onCreate()方法里使用此对象,因为还没有初始化,<span style="color:#008000;">可以使用线程解决,或者在一些监听器的实现类里面使用都没有问题。</span></span>  
  2.             mService = binder.getService();  
  3.             mBound = true;  
  4.         }  
  5.   
  6.         @Override  
  7.         public void onServiceDisconnected(ComponentName arg0) {  
  8.             mBound = false;  
  9.         }  
  10.     };  
  11. }  


   这里就是通过IBinder来得到LocalService对象,再去调用其Public方法。

方法三:使用Messenger

   上面的方法只能在同一个进程里才能用,如果要与另外一个进程的Service进行通信,则可以用Messenger。

    其实实现IPC的方式,还有AIDL,但推荐使用Messenger,有两点好处:

      1. 使用Messenger方式比使用AIDL的方式,实现起来要简单很多

      2. 使用Messenger时,所有从Activity传过来的消息都会排在一个队列里,不会同时请求Service,所以是线程安全的。如果你的程序就是要多线程去访问Service,就可以用AIDL,不然最好使用Messenger的方式。

  不过,其实Messenger底层用的就是AIDL实现的,看一下实现方式,先看Service的代码:

  

[java]  view plain copy
  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的代码:

[java]  view plain copy
  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发送消息,只要把代码反过来写就可以了。

你可能感兴趣的:(进程间通信 binder机制)