文档章节

Android中的IPC方式-Messenger --转载自肖老师博客160303

l
 lanou3g
发布于 2016/03/18 09:24
字数 987
阅读 10
收藏 0

原文地址:http://blog.csdn.net/xll712/article/details/50800698


  1. IPC简介

    IPC,即Inter-Process Communication进程间通信或者跨进程通信,是指两个进程之间进行数据交换的过程。
     android中的IPC方式有很多种,以下是几个简单的对比:

    这里写图片描述

    以上几种各有利弊,可能我们平时用的多的还是Bundle、文件共享、AIDL、ContentProvider这几种,对于Messenger这种方式很少用,而本章内容为了弥补这块的缺憾特写此贴,希望我们的知识又能扩展扩展,不好勿喷。

  2. Messenger

    Messenger可以翻译为信使,通过它可以在不同进程中传递Message对象,这时候可能会联想到咱们之前学的Handler对象,只不过Handler对象主要是进行线程间的数据通信的,而现在的Messenger则是进程间的通信的。

    Messenger是一种轻量级的IPC方案,它的底层实现是AIDL;它实际上是对AIDL的一种封装。

    下面是Messenger类的继承关系:

    这里写图片描述

    使用Messenger与之前使用的Handler类似,同样分为服务端和客户端。

    • 服务端进程
      拿Service举例,需要创建一个Messenger对象,用于接收Activity中发送过来的消息,服务端代码如下:

    public class MyService extends Service{

    private Messenger messenger;    private Handler handler;    @Override
    public void onCreate() {        // TODO Auto-generated method stub
        super.onCreate();
        handler = new Handler(new Handler.Callback() {            @Override
            public boolean handleMessage(Message msg) {                // TODO Auto-generated method stub
                //获取从Activity中发送过来的消息
                Bundle bundle = msg.getData();
                String str = bundle.getString("msg");
                System.out.println("----->>"+str);                //回执消息 即服务器收到消息之后可以反向给客户端发送消息
                Messenger servi = msg.replyTo;//获得从客户端传过来的信使对象Messenger

                Message reMsg = Message.obtain();
                Bundle bundle2 = new Bundle();
                bundle2.putString("msg", "消息回来了");
                reMsg.setData(bundle2);                try {
                    servi.send(reMsg);//向Activity中发送回执消息
                } catch (RemoteException e) {                    // TODO Auto-generated catch block
                    e.printStackTrace();
                }                return false;
            }
        });        //通过Handler对象创建一个Messenger对象
        messenger = new Messenger(handler);
    }    @Override
    public IBinder onBind(Intent intent) {        // TODO Auto-generated method stub
        //从Messenger对象中获取IBinder对象并返回
        return messenger.getBinder();
    }
}12345678910111213141516171819202122232425262728293031323334353637383940414243444546

备注:此处的replyTo主要是为了反向传输数据,即Service向ACtivity发送数据,Service中需要获得Activity创建的Messenger对象,用于消息的发送。

- 客户端进程
  相当于绑定服务的Activity对象,首先需要绑定服务端的Service,成功后用服务端返回的IBinder对象创建一个Messenger,通过这个Messenger就可以向服务端发送消息了。

public class MessengerActivity extends Activity{
    private Messenger messenger,replyMessenger;    private Handler handler;    private ServiceConnection conn = new ServiceConnection() {        @Override
        public void onServiceDisconnected(ComponentName name) {            // TODO Auto-generated method stub
        }        @Override
        public void onServiceConnected(ComponentName name, IBinder service) {            // TODO Auto-generated method stub
            //服务绑定成功之后 通过返回的IBinder对象创建Messenger对象 用于消息的发送
            messenger = new Messenger(service);
        }
    };    @Override
    protected void onCreate(Bundle savedInstanceState) {        // TODO Auto-generated method stub
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_b);

        handler = new Handler(new Handler.Callback() {            @Override
            public boolean handleMessage(Message msg) {                // TODO Auto-generated method stub
                //获取从Service中发送过来的数据
                Bundle bundle = msg.getData();
                String str = bundle.getString("msg");
                System.out.println("======>>>>"+str);                return false;
            }
        });        //创建一个Messenger对象 用于接收从Service中发送过来的消息
        replyMessenger = new Messenger(handler);        //启动服务 采用的是bindService的方式
        Intent intent = new Intent(this, MyService.class);
        bindService(intent, conn, Context.BIND_AUTO_CREATE);
    }    @Override
    protected void onDestroy() {        // TODO Auto-generated method stub
        super.onDestroy();        //解绑服务对象
        unbindService(conn);
    }    public void onMyClick(View v) {        if (messenger != null) {            //构建要发送的Message消息
            Message msg = Message.obtain();
            Bundle bundle = new Bundle();
            bundle.putString("msg", "hello word");
            msg.setData(bundle);

            msg.replyTo = replyMessenger;//如果想收到服务器向客户端发送过来的消息 必须把回执类传递给服务类

            try {
                messenger.send(msg);//发送消息 向存活在另一进程的service
            } catch (RemoteException e) {                // TODO Auto-generated catch block
                e.printStackTrace();
            }
        }
    }
}1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071

总结上述代码的工作原理,如下图所示:

这里写图片描述

注意:在AndroidManifest.xml文件中为了更能体现进程间的通信,可将Service注册为远程服务:即

<service android:name="com.example.exampledemo.MyService"
            android:process=":remote"></service>


© 著作权归作者所有

l
粉丝 0
博文 29
码字总数 33718
作品 0
海淀
私信 提问
Android 使用Messenger跨进程通信框架

一.通过Binder绑定形式的通信 上一篇说道Binder机制的通信框架,也说过Messenger的底层实现自AIDL,因此对于跨进程通信中,Messenger是一种比较高级的框架,可以说对于一个app开发者来说重要性...

IamOkay
2014/12/04
254
0
四大组件之Service(三)-Service的跨进程调用

版权声明:本文为博主原创文章,禁止转载,违者必究。 https://blog.csdn.net/anddlecn/article/details/51671035 第4节 远程调用 之前提到过:如果站在Service与触发Service运行的那个组件的...

anddlecn
2016/06/14
0
0
知识总结 插件化学习 Binder机制原理

Binder是android系统特有IPC方式,安卓平台中的各种服务交互基本都是Binder机制实现,理解和掌握Binder机制的实现原理可有效提升软件性能优化点,同时Binder机制的应用也是动态代理方式实现插...

常兴E站
2017/06/05
0
0
[翻译]Android Bound Services

一个bound service是一个client-server接口中的server端。一个bound service允许应用组件(比如activities)bind到它,发送请求,接收响应,甚至是执行进程间通信(IPC)。一个bound service在典...

WolfCS
2014/03/23
111
0
进程通信(IPC)之Messenger

bound服务是客户端-服务器模式的服务。 bound服务的创建方式之一: 创建一个支持绑定的服务时,你必须提供一个 IBinder,用作客户端和服务间进行通信的编程接口 使用Messenger定义该接口: ...

新根
2015/08/01
121
0

没有更多内容

加载失败,请刷新页面

加载更多

川普给埃尔多安和内堪尼亚胡的信

任性 https://twitter.com/netanyahu/status/1186647558401253377 https://edition.cnn.com/2019/10/16/politics/trump-erdogan-letter/index.htm...

Iridium
19分钟前
10
0
golang-mysql-原生

db.go package mainimport ("database/sql""time"_ "github.com/go-sql-driver/mysql")var (db *sql.DBdsn = "root:123456@tcp(127.0.0.1:3306)/test?charset=u......

李琼涛
48分钟前
5
0
编程作业20191021092341

1编写一个程序,把用分钟表示的时间转换成用小时和分钟表示的时 间。使用#define或const创建一个表示60的符号常量或const变量。通过while 循环让用户重复输入值,直到用户输入小于或等于0的值...

1李嘉焘1
48分钟前
7
0
Netty整合Protobuffer

现在我们都知道,rpc的三要素:IO模型,线程模型,然后就是数据交互模型,即我们说的序列化和反序列化,现在我们来看一下压缩比率最大的二进制序列化方式——Protobuffer,而且该方式是可以跨...

算法之名
53分钟前
18
0
如何用C++实现栈

栈的定义 栈(stack)又名堆栈,它是一种运算受限的线性表。限定仅在表尾进行插入和删除操作的线性表。这一端被称为栈顶,相对地,把另一端称为栈底。向一个栈插入新元素又称作进栈、入栈或压...

BWH_Steven
今天
5
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部