使用Messenger进行跨进程通信

最近在学习Android开发艺术探索这本书,看到Messenger的跨进程通信,觉得书上讲得不错,所以在这里做个总结。
Messenger的思路很简单,通过它可在不同进程传递Message对象,我们在Message中放入我们需要传递的数据,就可以轻松实现数据的进程间传递了。Messenger的底层是AIDL,它对AIDL做了封装,使我们可以更简单地进行进程间通信。

1.服务端进程

public class MessengerService extends Service {
private Handler handler = new Handler(){ 
   @Override    
public void handleMessage(Message msg) {
        if (msg.what==MSG){
            Toast.makeText(MessengerService.this, msg.getData().getString("msg"), Toast.LENGTH_SHORT).show(); 
       }  
  }
}; 
   private Messenger messenger = new Messenger(handler);
    @Override 
   public IBinder onBind(Intent intent) {
        // TODO: Return the communication channel to the service. 
       return messenger.getBinder();   
 }
}

看看这个服务端代码,我们使用了一个Handler处理客户端发送的Message,从Message取出我们所需的数据。
我们新建了一个Messenger对象,从构造参数我们可以看出它和handler相关联。我们在onBind()方法中返回了messenger中的binder对象。

由于我们处理的是跨进程通信,所以别忘了在Manifest文件中给Service加上process属性.

<service
    android:name=".MessengerService"
    android:process=":remote"/>

我们知道,一个应用的进程默认名称为包名,应用内的所有组件都运行在这个进程内,如果想让某个组件独立运行在另一个进程,那么则加上process属性,这里的“:remote”全程应该是包名:remote,省去了前面的包名。

2.客户端进程##

public class MainActivity extends AppCompatActivity {
    private Messenger messenger;
    public static final int MSG = 1; 
    private Button btn; 
   private ServiceConnection connection = new ServiceConnection(){
        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {
            messenger  = new Messenger(service); 
           Message message = Message.obtain(null,MSG);
            Bundle bundle  = new Bundle();
            bundle.putString("msg","helloworld");
            message.setData(bundle);  
          try { 
               messenger.send(message);
            } catch (RemoteException e) { 
               e.printStackTrace();
            }
        }        
       @Override 
       public void onServiceDisconnected(ComponentName name) { 
       } 
   }; 
   @Override 
   protected void onCreate(Bundle savedInstanceState) { 
       super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        btn = (Button) findViewById(R.id.bindService);
        btn.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
        Intent intent = new Intent(MainActivity.this,MessengerService.class);                bindService(intent,connection,BIND_AUTO_CREATE);
            } 
       });   
 } 
   @Override
    protected void onDestroy() { 
       unbindService(connection); 
       super.onDestroy(); 
   }
}

这个代码我们也比较熟悉了,跟平常bindService在客户端和服务端在同一进程的写法相似。区别在于使用了Messenger.
同样地我们在客户端也要新建一个Messenger对象,在构造参数里传入返回的binder对象。
我们在Message中放入我们要的数据,借助的是Bundle,最后用messenger.send(message);发送过去即可。

我们运行一下程序,可见服务端收到了Messenger发来的信息。

776440378629156779.png

在Messenger传输数据需要将数据放入Message.我们都知道,跨进程运输的数据需要实现序列化,所以Messenger和Message必定实现了Parcelable接口,看下源码

public final class Message implements Parcelable
public final class Messenger implements Parcelable

Message的载体有what,arg1,arg2,Bundle,object和replyTo,在同一进程内,obj很常用,但是在跨进程中就不行了,只有系统提供的Parcelable对象才可以通过它进行传输,我们自己定义的Parcelable对象是无法通过它传输的。但是我们还有Bundle,只要对象实现了Parcelable接口,利用bundle.putExtra(key,object)就能传递数据了,这里我们只演示传递String类型数据。

接下来我们再来看看服务端如何回复客户端发送的消息。
当服务端接收到客户端的消息时,我们也新建一个Messenger对象回复客户端,代码如下

//重点在这句话
Messenger messenger = msg.replyTo;
Message replyMessage = Message.obtain(null,REPLYMESSAGE);
Bundle bundle = new Bundle();
bundle.putString("reply","好的我知道了");
replyMessage.setData(bundle);

messenger对象由replyTo赋值。而replyTo来自客户端的Messenger,这样两者就建立联系。
再看看客户端的修改,我们新建了Handler对象和Mssenger对象。

private Handler getReplyHandler = new Handler(){
    @Override
    public void handleMessage(Message msg) {
        if(msg.what==REPLYMESSAGE){
            Toast.makeText(MainActivity.this,msg.getData().getString("reply"),Toast.LENGTH_SHORT).show();        
}
    }
};
private Messenger getReplyMessenger = new Messenger(getReplyHandler);

然后我们需要把接收服务端回复的Messenger通过Message的replyTo参数传递给服务端,这句话有点难理解,多看几遍。

private ServiceConnection connection = new ServiceConnection(){
    @Override
    public void onServiceConnected(ComponentName name, IBinder service) {
        messenger  = new Messenger(service);
        Message message = Message.obtain(null,MSG);
        Bundle bundle  = new Bundle();
        bundle.putString("msg","成功进行跨进程通信!");
        message.setData(bundle);
        //这句话与服务端的Messenger messenger = msg.replyTo;相对应,缺一不可
        message.replyTo = getReplyMessenger;
        try {
            messenger.send(message);
        } catch (RemoteException e) {
            e.printStackTrace(); 
       }
    }

再运行一下程序

230912919011775556.png

最后再放上一张Messenger的工作原理,找不到书上的原图,只能自己画了。


Messenger.jpg
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,384评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,845评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,148评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,640评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,731评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,712评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,703评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,473评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,915评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,227评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,384评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,063评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,706评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,302评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,531评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,321评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,248评论 2 352

推荐阅读更多精彩内容