ZeroMQ(java)中的数据流SessionBase与SocketBase
前面的文章中已经比较的清楚了ZeroMQ(java)中如何在底层处理IO,
通过StreamEngine对象来维护SelectableChannel对象以及IO的事件回调,然后通过Poller对象来维护Selector对象,然后用IOObject对象来具体的管理SelectableChannel对象在Poller上面的注册,以及事件回调,他们之间的关系可以用下面的图形来简单的描述一下:
对于接收到的数据,首先由StreamEngine进行处理,其实它会调用内部的decoder将字节数据转化为Msg对象,然后再交给上层的对象,
其实这里的上层对象也就是Session对象,每一个StreamEngine对象都有一个自己的Session对象,然后Session对象收到下面传上来的数据之后,再会通过Pipe,将数据发送到其更上层的Socket对象,
然后接下来的数据处理就交由用户的代码来处理了。。。
对于刚刚提到的对象之间的层次,用下面的图形来描述吧:
这张图应该还算刻画的比较直接了吧,底层数据通信部分负责从channel接收数据和发送二进制的数据,然后又Decoder以及Encoder来负责字节数据与Msg之间的转化。。。。
我们在ZMQ中会看到很多种类的Socket,例如Req,Dealer,Router啥的,他们都继承自SocketBase类型,每一种类型都有自己的Session,都继承自SessionBase类型。。。
好了,那么接下来先来看看SessionBase类型吧,每一个StreamEngine对象都有一个Session对象与之关联,他们是一对一的关系,先来看看它的一些重要的属性定义吧:
- private boolean connect; //是否需要连接,如果是fasle的话,那么表示是listener创建的连接
- private Pipe pipe; //与socket进行通信的pipe
- private final Set<Pipe> terminating_pipes;
- //如果是true的话,表示还有message在pipe里面没有执行
- private boolean incomplete_in;
- //如果是true的话,表示停止发送数据到网络中
- private boolean pending;
- private IEngine engine; //绑定到这个session上面的底层的通信
- private SocketBase socket; //当前session所属的socket
- private IOThread io_thread; //当前session关联的IO线程,底层的io将会加入到这个IO线程
- private static int linger_timer_id = 0x20;
- // True is linger timer is running.
- private boolean has_linger_timer; //有超时事件的注册?
- private boolean identity_sent; //如果是true的话,表示标志已经发送了
- private boolean identity_received; //表示标志已经接收了
- private final Address addr; //连接的地址
- private IOObject io_object; //关联的IO对象,IEngine里的io对象
具体这些属性的是干嘛的,上面的注释基本上都已经给出来的吧,这里比较重要的属性是:
pipe,它用于与上面的Socket进行通信,当下层有数据被解析出来以后,会通过pipe将msg发送给上层的socket,具体pipe的过程,前面的文章已经说过了。。。
IEngine,底层数据通信的StreamEngine对象的引用,这个重要性就不说了吧,。。。
IOThread对象,这个是当前Session对象将会依赖的IO线程,也就是发给session的命令都会被这个IO线程的mailbox接收到,从而在这个线程中执行命令,嗯。。。重要吧。。。
另外还有一些标志位什么的。。。
好了,这里就不细说,来看看一些重要的方法吧:
- //与pipe关联上,这里其实主要是为了将当前pipe的事件回调设置为当前对象
- public void attach_pipe(Pipe pipe_) {
- assert (!is_terminating ());
- assert (pipe == null);
- assert (pipe_ != null);
- pipe = pipe_; //保存当前的pipe
- pipe.set_event_sink (this); //将当前的pipe的事件回调社设置为当前
- }
用于关联pipe对象,这里可以看到将pipe的事件回到设置成了当前session对象。。那么来看看这些事件回调方法是怎么处理的吧:
- //当有数据传给当前的pipe的时候,其实也就是pipe的对面socket那边发送数据给这里了,然后发送pipe可读的命令,那么表示有数据需要通过底层的engine发送出去了
- public void read_activated(Pipe pipe_) {
- // Skip activating if we're detaching this pipe
- if (pipe != pipe_) {
- assert (terminating_pipes.contains (pipe_));
- return;
- }
- if (engine != null)
- engine.activate_out (); //激活底层engine的channel的写事件
- else
- pipe.check_read ();
- }
- //表示可以发送数据到pipe了,那么表示需要到底层的engine接收数据了
- public void write_activated (Pipe pipe_) {
- // Skip activating if we're detaching this pipe
- if (pipe != pipe_) {
- assert (terminating_pipes.contains (pipe_));
- return;
- }
- if (engine != null)
- engine.activate_in (); //激活底层的engin的channel上面的读取事件,也就是通知底层的engin应该从网络接收数据了
- }
这里先是pipe可以读的时候的事件回调方法,这个处理很简单吧,直接激活底层StreamEngine的channle在poller上注册写事件,那么当底层channel可以写数据的时候,就会从当前session的pipe里去读取数据,然后发送出去。。
第二个方法是当pipe可以写的时候,这个其实就是直接注册channel的读取事件,那么当channel就会去接收数据,最后这些他们都会通过pipe发送给上层的socket对象。。。到此上面那张图的整个运行情况应该都很清楚了吧。。。
那么Session的最为关键的地方也就差不多了。。。还有一些细节,以后有需要的话再介绍吧。。。
好了接下来来看看SocketBase这个类型吧,前面已经说到了ZMQ中所有的Socket类型多继承自这个类型,可见他的重要性。。。先来看看它的一些重要的属性定义吧:
- private final Map<String, Own> endpoints; //这里保存所有打开的endpoint,这里key是连接地址,value其实是Session对象
- private final Map<String, Pipe> inprocs; //IPC通信方法
- // Used to check whether the object is a socket.
- private int tag; //标志位,用于判断当前对象是否是socket类型的
- private boolean ctx_terminated; //如果是true,表示关联的context已经停止了
- private boolean destroyed; //如果是true的话,表示当前socket对象已经被命令销毁了
- private final Mailbox mailbox; //邮箱,用于接收别的地方发送过来的命令
- private final List<Pipe> pipes; //当前所有关联的pipe,这些pipe是用于与当前socket的所有session进行数据通信的
- private Poller poller; //poller对象
- private SelectableChannel handle; //这个是mailbox的handler
- private long last_tsc; //上一次执行命令的时间
- private int ticks; //上次执行完命令之后,收到的message
- private boolean rcvmore; //接着还有message要接收
- private SocketBase monitor_socket; //用于监控的socket
- private int monitor_events;
- protected ValueReference<Integer> errno;
这里比较重要的有,:
enpoints,用于保存所有的session与其连接地址的,
pipes,保存所有与底层的session关联的pipe,这里Socket与Sesssion之间的关系是一对多的...
mailbox,socket也有自己的mailbox,不用依附于IOThread对象,不过这里有个坑,Socket类型的对象有自己的mailbox,不用依附于IO线程,并不意味着它就有自己的线程,因为它直接依赖于用户线程,依赖于用户代码...也就是说mailbox里面的命令的执行都是在用户线程中搞定的...呵呵,刚开始这个地方还纠结了很久...
来看看它的pipe的事件回调吧:
- //当pipe可以读的时候需要执行的回调方法,表示底层的StreamEngin有数据发送到这里了
- public void read_activated (Pipe pipe_) {
- xread_activated(pipe_); //调用子类的方法来具体处理这些数据
- }
- //当pipe可以写的时候执行的回调
- public void write_activated (Pipe pipe_) {
- xwrite_activated (pipe_);
- }
其实这里没有太多的内容,因为都是在具体的子类中完成的,不过到这里整个ZeroMQ(java)中数据是怎么进行流动的就算已经很清楚了...当然,有一些细节性的东西还没有列出来..
好了,到现在为止,ZeroMQ(java)中就还剩下具体的socket类型的运行以及编码方式两个大的地方没有分析了...
ZeroMQ(java)中的数据流SessionBase与SocketBase的更多相关文章
- ZeroMQ(JAVA)中的数据流,SessionBase与SocketBase
前面的文章中已经比较的清楚了ZeroMQ(java)中如何在底层处理IO, 通过StreamEngine对象来维护SelectableChannel对象以及IO的事件回调,然后通过Poller对象来维 ...
- ZeroMQ(java)中监控Socket
基本上ZeroMQ(java)中基本的代码都算是过了一遍了吧,不过觉得它在日志这一块貌似基本没有做什么工作,也就是我们通过日志来知道ZeroMQ都发生了什么事情.. 而且由于ZeroMQ中将连接的建立 ...
- ZeroMQ(java)中组件间数据传输(Pipe的实现)
在ZeroMQ(java)中,整个IO的处理流程都是分层来进行的,当然处于最下端的肯定是前面介绍过的poller以及StreamEngin了....涉及到上层的话就还有session,以及socket ...
- Java中I/O流之数据流
Java 中的数据流: 对于某问题:将一个 long 类型的数据写到文件中,有办法吗? 转字符串 → 通过 getbytes() 写进去,费劲,而且在此过程中 long 类型的数需要不断地转换. ...
- java中数据流的简单介绍
java中的I/O操作主要是基于数据流进行操作的,数据流表示了字符或者字节的流动序列. java.io是数据流操作的主要软件包 java.nio是对块传输进行的支持 数据流基本概念 “流是磁盘或其它外 ...
- 【转】输入/输出流 - 深入理解Java中的流 (Stream)
基于流的数据读写,太抽象了,什么叫基于流,什么是流?Hadoop是Java语言写的,所以想理解好Hadoop的Streaming Data Access,还得从Java流机制入手.流机制也是JAVA及 ...
- JAVA中管道通讯(线程间通讯)例子
Java I/O系统是建立在数据流概念之上的,而在UNIX/Linux中有一个类似的概念,就是管道,它具有将一个程序的输出当作另一个程序的输入的能力.在Java中,可以使用管道流进行线程之间的通信,输 ...
- java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET
java中的io系统详解 - ilibaba的专栏 - 博客频道 - CSDN.NET 亲,“社区之星”已经一周岁了! 社区福利快来领取免费参加MDCC大会机会哦 Tag功能介绍—我们 ...
- java中hashCode方法与equals方法的用法总结
首先,想要明白hashCode的作用,必须要先知道Java中的集合. 总的来说,Java中的集合(Collection)有两类,一类是List,再有一类是Set. 前者集合内的元素是有序的,元素可以重 ...
随机推荐
- Boostrap(1)
1.简介 Bootstrap 是一个用于快速开发 Web 应用程序和网站的前端框架.Bootstrap 是基于 HTML.CSS.JAVASCRIPT 的,可以认为bootstrap就是一个样式库. ...
- 第三章:模块加载系统(requirejs)
任何一门语言在大规模应用阶段,必然要经历拆分模块的过程.便于维护与团队协作,与java走的最近的dojo率先引入加载器,早期的加载器都是同步的,使用document.write与同步Ajax请求实现. ...
- 【前端开发系列】—— CSS3属性选择器总结
想想自己为什么要学CSS,作为一个开发过前端的人员来说,调试一个图片花了半天的时间,最后发现分隔符用错了,实在是一件很丢人的事情.因此,痛下决心来学习CSS,最近一周也会更新下相关的学习笔记. CSS ...
- XCode7继续用http协议解决办法
昨天被苹果放鸽子也没升级iOS9,今天升级了Xcode7,同时手机升级了iOS9,发现项目报错,查了查才知道是iOS9不支持不安全的http传输协议,让用https协议,这根本就不x现实,,服务端根本 ...
- Java-日期转换
如下: package 时间日期类; import java.text.SimpleDateFormat; import java.util.Date; public class 日期格式转换 { / ...
- 【转】从Go、Swift语言出发
Google于2009年第一次提出了Go的构思,Facebook在去年春天引入了Hack,随后不久Apple也发布了其Swift语言. 在战争中,胜利者写历史书:在科技中,赢的公司都在写编程语言.互联 ...
- Java编程思想学习(六) 多态
1.Java语言的三大特性:继承.封装和多态. 继承:复用类的一种方法,可以简省很多代码: 封装:通过合并特征和行为来创建新的数据类型.[这种“数据类型”跟Java本身提供的8大“基本数据类型”的地位 ...
- 【poj2342】 Anniversary party
http://poj.org/problem?id=2342 (题目链接) 题意 没有上司的舞会... Solution 树形dp入门题. dp[i][1]表示第i个节点的子树当节点i去时的最大值,d ...
- [NOIP2011] 提高组 洛谷P1315 观光公交
题目描述 风景迷人的小城Y 市,拥有n 个美丽的景点.由于慕名而来的游客越来越多,Y 市特意安排了一辆观光公交车,为游客提供更便捷的交通服务.观光公交车在第 0 分钟出现在 1号景点,随后依次前往 2 ...
- RIP、OSPF、BGP、动态路由选路协议、自治域AS
相关学习资料 tcp-ip详解卷1:协议.pdf http://www.rfc-editor.org/rfc/rfc1058.txt http://www.rfc-editor.org/rfc/rfc ...