您的位置:首页 > 移动开发 > Android开发

Messenger解析(和Handler、IBinder、Message的关系)

2015-10-11 17:15 429 查看
为了更好的理解bound service利用Messenger来夸进程服务,这里先写一篇文章简单分析下Messenger、Handler、Message、Binder的关系。

先来看看Messenger官方的定义:
Reference to a Handler, which others can use to send messages to it.This allows for the implementation of message-based communication acrossprocesses, by creating a Messenger pointing to a Handler in one process,and handing that Messenger to another process.
Messenger关联一个可以发送消息的Handler。通过指定一个Handler对象创建一个Messenger可以实现基于消息的夸进程通信。

你可以把Messenger理解成一个送信人,他唯一的职责就是送信,即:send(Message msg),来看看send(Message msg)的实现:

发送消息的方法:

public void send(Message message) throws RemoteException {
mTarget.send(message);
}

焦点关注到mTarget,看看它是什么东西?send方法又是怎么回事?和handler有关系?
其实Messenger的源码很简单,只有一个成员变量:mTarget(IMessenger的一个实例),核心就是它了,mTarget在构造方法中产生。
两个构造方法:

public Messenger(Handler target) {
mTarget = target.getIMessenger();
}
public Messenger(IBinder target) {
mTarget = IMessenger.Stub.asInterface(target);
}

先来说说第一个构造方法(第二个构造方法也许你有些疑惑先放一边,后面解释)。
我们发现,Handler有个getIMessenger()。来看看源码实现:

final IMessenger getIMessenger() {
synchronized (mQueue) {
if (mMessenger != null) {
return mMessenger;
}
mMessenger = new MessengerImpl();
return mMessenger;
}
}

private final class MessengerImpl extends IMessenger.Stub {
public void send(Message msg) {
msg.sendingUid = Binder.getCallingUid();
Handler.this.sendMessage(msg);
}
}


看到这里,我们发现 mTarget 是一个在Handler中的实现的IMessenger实例(IMessenger.Stub 是IMessenger的实现类,里面实现了n多方法,这里不做深究),里面实现了send方法,这个方法调用了Handler的sendMessage方法。看到这里豁然开朗了吧。 mTarget.send(message);其实最终就是Handler。
看到这里,Messenger和Handler的关系很明显了,即:Messenger利用Handler发送消息。
上面还有个疑问,第二个构造方法是怎么回事? 
public Messenger(IBinder target) {
mTarget = IMessenger.Stub.asInterface(target);
}

Create a Messenger from a raw IBinder, which had previously been retrieved getBinder。

这是官方解释,即,通过IBinder创建一个Messenger,这个IBinder必须是之前存在的从getBinder()方法中获取的对象

public IBinder getBinder() {
return mTarget.asBinder();
}

我们把这两个方法放在一起 ,你会发现mTarget(IMessenger)和IBinder存在转化关系,即IMessenger.Stub已经帮我们实现了这种转化方法 asInterface(),IBinder-》mTarget;asBinder(),mTarget-》IBinder。所以结论是:IBinder可以关联到一个Handler,通过一个Messenger的getBinder()获取的IBinder可以创建一个关联相同Handler的Messager。
 Messenger和Message的关系?
 Message 有个成员:replyTo,它是一个Messenger的实例,它有什么作用呢?我们来看一个例子,在这个例子中我们实现两个线程利用Messenger通信。
一个类:MessengerThread
public class MessengerThread extends Thread {

private Handler inComingHandler = new Handler() {
@Override
public void dispatchMessage(Message msg) {

Log.d("MessengerThread",msg.obj.toString());

try {
Message reMsg = Message.obtain();
reMsg.replyTo = MessengerThread.this.getMessenger();
reMsg.obj = "reply:hello ,am "+getName();
msg.replyTo.send(reMsg);
} catch (RemoteException e) {
e.printStackTrace();
}

}
};;
private Messenger messenger;
private MessengerThread another;

public MessengerThread(String name) {
super(name);
messenger = new Messenger(inComingHandler);

}

public void setAnother(MessengerThread another){
this.another = another;
}
public Messenger getMessenger(){
return messenger;
}

@Override
public void run() {
if (another != null){
Message  msg = Message.obtain();
msg.replyTo = another.getMessenger();
msg.obj = "hello i am "+getName();
try {
another.getMessenger().send(msg);
} catch (RemoteException e) {
e.printStackTrace();
}
}
super.run();
}
}

测试代码:

MessengerThread messengerThreadA = new MessengerThread("A");
MessengerThread messengerThreadB = new MessengerThread("B");
messengerThreadA.setAnother(messengerThreadB);
messengerThreadB.setAnother(messengerThreadA);
messengerThreadB.start();
messengerThreadA.start();

输出:

949-6949/com.demo.liuguangli.messenger D/MessengerThread﹕ reply:hello ,am A
10-11 16:22:36.573    6949-6949/com.demo.liuguangli.messenger D/MessengerThread﹕ reply:hello ,am B
10-11 16:22:36.573    6949-6949/com.demo.liuguangli.messenger D/MessengerThread﹕ reply:hello ,am A
10-11 16:22:36.574    6949-6949/com.demo.liuguangli.messenger D/MessengerThread﹕ reply:hello ,am B

demo地址:https://github.com/liuguangli/MessengerDemo
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息