转载: http://blog.csdn.net/yanbober

1 背景

这个知识点是个low货,刚开始其实想在之前一篇文章《Android异步消息处理机制详解及源码分析》一文中作为一个知识点分析的,但是想了又想又觉得该放在后面进程间通信分析时再分析。然并卵,还是单独拿出来写一篇分析一下吧。

提到Message和Handler其实大家都很熟悉,但是说到Messenger估计有些人还是不太常用的,更有甚者都能把Messenger拼写错误为Messager,以为是Message加了个r,当然,网络上对于Messenger的文章现在也很多了,但是个人分析总结总归是个人的。哈哈,不扯淡了,言归正传。

Messenger实现了IPC通信,其真实原理也是使用了AIDL进行通信,但是和直接使用AIDL不同的是Messenger利用了Handler处理通信,所以它是线程安全的(不支持并发处理);而我们平时用的AIDL是非线程安全的(支持并发处理)。所以大多数时候我们应用中是不需要处理夸进程并发处理通信的,所以这时选择Messenger会比AIDL更加容易操作。

【工匠若水 http://blog.csdn.net/yanbober 转载请注明出处。点我开始Android技术交流】

2 基础实例

分析源码之前我们先来看一个Demo例子,其核心逻辑就是客户端进程client发送一个消息到服务端进程remote,服务端进程收到消息后做完处理再回发一个消息到客户端client,整个过程采用了Messenger信使和Handler来实现。具体如下:

工程的Android管理文件:

<?xml version="1.0" encoding="utf-8"?>
<manifest xmlns:android="http://schemas.android.com/apk/res/android"package="com.example.yanbo.myapplication" ><application
        android:allowBackup="true"android:icon="@mipmap/ic_launcher"android:label="@string/app_name"android:theme="@style/AppTheme" ><activity
            android:name=".MainActivity"android:label="@string/app_name" ><intent-filter><action android:name="android.intent.action.MAIN" /><category android:name="android.intent.category.LAUNCHER" /></intent-filter></activity><!-- ":remote"表示在应用程序里,当需要该service时,会自动创建新的进程"remote"表示创建全局进程,不同的应用程序共享该进程 --><service android:name=".RemoteService"android:process=":remote"><intent-filter><action android:name="com.remote.RemoteService"/></intent-filter></service></application></manifest>
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30

工程中的一个独立进程服务端remote源码:

/*** 另一个进程中的Service*/
public class RemoteService extends Service {public static final int MSG_TAG_REMOTE = 0x110;public static final int MSG_TAG_CLIENT = 0x111;private Messenger mRemoteMessenger;private RemoteHandler mRemoteHandler;private int mCounter = 0;@Overridepublic void onCreate() {super.onCreate();mRemoteHandler = new RemoteHandler();//实例化一个Messenger传入当前HandlermRemoteMessenger = new Messenger(mRemoteHandler);}@Overridepublic IBinder onBind(Intent intent) {return (mRemoteMessenger == null) ? null : mRemoteMessenger.getBinder();}private class RemoteHandler extends Handler {@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case MSG_TAG_REMOTE://为了把消息回传给client端,所以获取client端设置的MessengerMessenger clientMessenger = msg.replyTo;if (clientMessenger != null) {try {//注意obtain第一个参数,前面文章有解释,因为参数target不可序列化clientMessenger.send(Message.obtain(null, MSG_TAG_CLIENT, ++mCounter, 0));} catch (RemoteException e) {e.printStackTrace();}}break;default:super.handleMessage(msg);break;}}}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48

工程中的一个独立进程客户端client代码:

public class MainActivity extends Activity {private TextView mTextView;private Messenger mRemoteMessenger = null;private Messenger mClientMessenger;private ClientHandler mClientHandler;@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);mTextView = (TextView) findViewById(R.id.content_show);//mClientHandler = new ClientHandler();mClientMessenger = new Messenger(mClientHandler);bindService(new Intent(this, RemoteService.class), connection, Context.BIND_AUTO_CREATE);}@Overrideprotected void onDestroy() {super.onDestroy();unbindService(connection);}private ServiceConnection connection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {mRemoteMessenger = new Messenger(service);//注意obtain第一个参数,前面文章有解释Message message = Message.obtain(null, RemoteService.MSG_TAG_REMOTE);message.replyTo = mClientMessenger;try {mRemoteMessenger.send(message);} catch (RemoteException e) {e.printStackTrace();}}@Overridepublic void onServiceDisconnected(ComponentName name) {}};private class ClientHandler extends Handler {@Overridepublic void handleMessage(Message msg) {switch (msg.what) {case RemoteService.MSG_TAG_CLIENT:if (mTextView != null) {mTextView.setText(msg.arg1+"");}break;default:super.handleMessage(msg);break;}}}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 44
  • 45
  • 46
  • 47
  • 48
  • 49
  • 50
  • 51
  • 52
  • 53
  • 54
  • 55
  • 56
  • 57
  • 58

看着了吧,这就是一个超级简单的Messenger使用场景,具体过程比较形象的描述如下图:

相信有了这幅图就不需要再解释啥了吧,这个也够明白了。

【工匠若水 http://blog.csdn.net/yanbober 转载请注明出处。点我开始Android技术交流】

3 Messenger源码浅析

通过上面的实例明显可以看出,在不考虑并发的情况下,Messenger相比AIDL无论从代码量、工程结构、复杂度等上都更加胜出一筹。既然这么好用的东东,那就来看看他的源码吧,如下我们先通观一下Messenger类的整个核心代码,然后再细说。如下所示:

/*** 关联Handler进行跨进程收发消息的信使管理桥梁类* 可以看见Messenger就是一个信使,就是一个Object*/
public final class Messenger implements Parcelable {//其实就是远程的MessengerService的AIDL接口private final IMessenger mTarget;//创建一个指向target Handler的Messenger,然后调运Messenger的send就像Handler的sendMessagepublic Messenger(Handler target) {mTarget = target.getIMessenger();}//跨进程发送消息,通常用Message.obtain()填充message参数,也可以自己newpublic void send(Message message) throws RemoteException {mTarget.send(message);}//获得Messenger的Binder,一般用在remote端获取返回public IBinder getBinder() {return mTarget.asBinder();}//如果两个Messenger相等则表明指向了相同的Handlerpublic boolean equals(Object otherObj) {if (otherObj == null) {return false;}try {return mTarget.asBinder().equals(((Messenger)otherObj).mTarget.asBinder());} catch (ClassCastException e) {}return false;}......//获取getBinder相同的Messenger对象,一般用在client端获取public Messenger(IBinder target) {mTarget = IMessenger.Stub.asInterface(target);}
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 24
  • 25
  • 26
  • 27
  • 28
  • 29
  • 30
  • 31
  • 32
  • 33
  • 34
  • 35
  • 36
  • 37
  • 38
  • 39
  • 40
  • 41
  • 42
  • 43

通过上面全局预览Messenger类及上面的实例使用相信你一定注意到了Messenger有两个构造函数,分别是public Messenger(Handler target)和public Messenger(IBinder target),很明显你已经知道了,参数为Handler的是远程进程实例方法,而参数为IBinder为客户端进程的实例方法。既然这样那我们就先从服务端获取Messenger对象的构造函数public Messenger(Handler target)说起吧,可以看见代码如下:

public Messenger(Handler target) {mTarget = target.getIMessenger();
}
  • 1
  • 2
  • 3
  • 1
  • 2
  • 3

该构造函数调运了Handler的getIMessenger方法,这个方法在Handler中源码如下:

public class Handler {......//其实对于一个Handler对象来说getIMessenger得到的Messenger是一个单例模式对象final IMessenger getIMessenger() {synchronized (mQueue) {if (mMessenger != null) {return mMessenger;}//单例模式得到Messenger实现类MessengerImpl对象mMessenger = new MessengerImpl();return mMessenger;}}//可以看见这其实是Messenger的AIDL实现private final class MessengerImpl extends IMessenger.Stub {public void send(Message msg) {//send方法真正调运了Handler的sendMessage实现发送消息Handler.this.sendMessage(msg);}}......
}
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 10
  • 11
  • 12
  • 13
  • 14
  • 15
  • 16
  • 17
  • 18
  • 19
  • 20
  • 21
  • 22
  • 23

可以看见,getIMessenger对于每一个Handler对象来说是单例的对象,而且这个IMessenger对象的实现类是MessengerImpl,也可以看见MessengerImpl又是IMessenger.Stub的实现类,这个IMessenger.Stub其实就是AIDL文件通过aapt自动生成在我们gen或者build目录下的服务端接口子类而已。那既然这么说了我们就来确认下吧,看下面这个AIDL文件(frameworks/base/core/Java/android/os/IMessenger.aidl):

package android.os;  import android.os.Message;  /** @hide */
oneway interface IMessenger {  //可以看见,上面的MessengerImpl就实现了Messenger远程的send接口void send(in Message msg);
}  
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9
  • 1
  • 2
  • 3
  • 4
  • 5
  • 6
  • 7
  • 8
  • 9

这下明白了吧,Messenger类中的mTarget其实就是一个Handler中单例的IMessenger远程IPC接口MessengerImpl。

紧接着我们看下Service中的onBind实现,其调运了Messenger的getBinder方法,这个方法源码如下:

public IBinder getBinder() {return mTarget.asBinder();
}
  • 1
  • 2
  • 3
  • 1
  • 2
  • 3

可以看见,其实asBinder返回的就是this,也就是自己,也就是把Service中的Messenger通过onBind方法返回给客户端。

接着我们暂时回到上面实例的客户端代码,可以发现,客户端进程首要任务就是通过与远程进程Service绑定然后获取远程Messenger对象实例,其用的构造函数如下:

public Messenger(IBinder target) {mTarget = IMessenger.Stub.asInterface(target);
}
  • 1
  • 2
  • 3
  • 1
  • 2
  • 3

握草,这不就是我们平时获取服务端对象的实现方法么?是的,就这样客户端就拿到了服务端的Messenger对象,看着好像客户端新new了一个对象似的,其实不是的。

然后就是发送过程了,发送过程的消息首先是可序列化的,然后通过Messenger的send发送,而这个send方法上面我们也分析过了,其实现就是通过Handler的sendMessage方法来发送的,所以不再多解释,又回归到了之前分析的Handler与Message过程。

总体可以发现,其实Messenger没啥高端的,坦白说其实就是对AIDL的二次封装后结合Handler让用户用起来更加便捷而已。

整个Messenger原理如下:

Android应用进程间通信之Messenger信使使用及源码浅析相关推荐

  1. Android应用进程间通信之Messenger信使使用及源码浅析,Android插件化入门指南

    } } break; default: super.handleMessage(msg); break; } } } } 工程中的一个独立进程客户端client代码: public class Mai ...

  2. Android开发之Theme、Style探索及源码浅析

    1 背景 前段时间群里有伙伴问到了关于Android开发中Theme与Style的问题,当然,这类东西在网上随便一搜一大把模板,所以关于怎么用的问题我想这里也就不做太多的说明了,我们这里把重点放在理解 ...

  3. Android Loader机制全面详解及源码浅析

    原文出处:csdn@工匠若水,http://blog.csdn.net/yanbober/article/details/48861457 一.概述 在Android中任何耗时的操作都不能放在UI主线 ...

  4. Android 经典示例,初学者的绝好源码资料

    2019独角兽企业重金招聘Python工程师标准>>> Android 经典示例,初学者的绝好源码资料 附上源码: 转载:http://www.adobex.com/android/ ...

  5. Android图片爬虫,看妹纸神器项目源码,ListView单张图片查看

    Android图片爬虫,看妹纸神器项目源码 刚自学完安卓,国庆7七天花了6天纯手写入门级app,从集成图片爬虫到整个项目结束真是一步一个坑. 整个项目没有用框架,都是手写纯属练手,项目中主要用到的技术 ...

  6. (仿微信Android)IM聊天+抢红包+直播+朋友圈源码发布了

    (仿微信Android)IM聊天+抢红包+直播+朋友圈源码发布了 功能概览: IM聊天 单聊/群聊/聊天室--基于环信sdk 红包功能: 1.一对一红包 2.群红包(抢红包.答题红包.专属红包) 3. ...

  7. 安卓Android与H5双向交互MathJax展示数学公式(源码+解析)

    安卓Android与H5双向交互MathJax展示数学公式(源码+解析) 博主就今天周五又做了个需求(安卓Android与H5交互),原来上线的功能是服务器配置过来的学习报告(一个H5页面)并提供原始 ...

  8. 超级应用 - 免费应用内测托管平台|APP应用分发平台|iOS应用分发|Android应用分发|免费应用内测托管平台 源码下载

    网站标题:超级应用 - 免费应用内测托管平台|APP应用分发平台|iOS应用分发|Android应用分发|免费应用内测托管平台源码下载 网站关键词:超级应用(www.awwjd.com) 为您提供测试 ...

  9. Android开发应用案例——简易计算器(附完整源码)

    Android开发-AS学习(一) Android开发-AS学习(二) 使用android studio开发简易计算器app(完整源码可在博主资源中自行下载) 最终效果: 开发步骤: 创建一个名为ca ...

最新文章

  1. HarmonyOS 修改App的默认加载的界面
  2. LAMP 全功能编译安装 for CentOS6.3笔记(更新)
  3. STM32速度---网页讲解
  4. 分布式架构的负载均衡算法
  5. 【操作系统】信号量解决经典同步问题
  6. 南充一中计算机机房被盗,成都理工大学与南充市第一中学共建优质生源基地
  7. 两个维护 提升三服务器,王莉霞:以整改成效践行“两个维护”立足“事要解决”抓好“三访结合”...
  8. android 添加ga_android开发步步为营之70:android接入Google Analytics总结
  9. HttpContext.Current.Cache在控制台下不工作
  10. SVN Description : The working copy is locked due to a previous error.
  11. 偏微分方程的数值解(二): 一维状态空间的偏微分方程的 MATLAB 解法
  12. 多选题spss相关分析_spss多选题的录入及分析
  13. 许鹏:从零开始学习,Apache Spark源码走读(三)
  14. H5标签input标签上传文件(图片)
  15. ARMv8基础架构之内存屏障(Memory Barriers)
  16. 《庖丁解牛Linux内核》笔记之:调用堆栈
  17. c语言编程 菲薄拉,C语言设计模式-封装-继承-多态
  18. C++ 的一些免费库
  19. support information
  20. itpub上的ORACLE之常用FAQ V1.0(转)

热门文章

  1. html语言保存格式化,HTML 文本格式化
  2. GeForce RTX™ 2070 MINI ITX 8G
  3. 常用牛人主页链接(计算机视觉、模式识别、机器学习相关方向,陆续更新。。。。)
  4. 苹果se2_苹果SE2已经被替代,配置A13的iPhone 9算不算大惊喜?
  5. 文明6游戏 linux,《文明6》正式支持Linux/SteamOS 但不支持Intel核显和A卡
  6. 01 NCC常用动作脚本
  7. 第06周:吴恩达 Andrew Ng 机器学习
  8. c语言中if语句嵌套的作用,C语言中if—else语句的嵌套分析
  9. 卖货文案一:激发行动欲望
  10. 故障电弧探测器在某医院项目上的应用