参考资料:

了解 Java NIO 的 Reactor 模型,大神 Doug Lea 的 PPT Scalable IO in Java 必看:http://gee.cs.oswego.edu/dl/cpjslides/nio.pdf

示例代码:

https://github.com/gordonklg/study,socket module

A. 单线程版

Reactor 相当于一个中央事件收集分发器。一方面,Reactor 通过 Selector 可以收到已经准备完毕的事件通知,另一方面,Reactor 将事件发送给对应的 Handler 处理。

对于 NIO 服务端,建立连接与数据传输是通过不同类型的 Channel 处理的。ServerSocketChannel 用来处理连接建立请求,其 accept 方法创建出的 SocketChannel 用来处理与客户端的数据传输。多数情况下,服务端会有一个 ServerSocketChannel 以及数量与已连接客户端总数一致的 SocketChannel。在单线程版 Reactor 模型中,所有的 Channel 都会注册到 Reactor 的 Selector 上,由 Reactor 的事件循环代码分发(dispatch)事件。

Reactor 将事件发送给对应的 Handler 处理,acceptor 可以看作一个特殊的 Handler,用于处理连接建立请求。而每个具体的连接(对应 SocketChannel)对应一个 Handler 实例,该 Handler 实例负责读取数据、解码、执行业务逻辑、编码以及发送数据给客户端。显然,Handler 是线程安全的。

nio.pdf 文件中包含了单线程版几乎所有的源码,自己只需要实现几个简单方法即可。我额外增加了一些调试日志用于观察系统运行情况。

gordon.study.socket.nio.reactor.singlethread.Reactor.java

public class Reactor implements Runnable {

    final Selector selector;

    final ServerSocketChannel serverSocket;

    public Reactor(int port) throws IOException {
selector = Selector.open();
serverSocket = ServerSocketChannel.open();
serverSocket.socket().bind(new InetSocketAddress(port));
serverSocket.configureBlocking(false);
SelectionKey sk = serverSocket.register(selector, SelectionKey.OP_ACCEPT);
sk.attach(new Acceptor());
} @Override
public void run() {
try {
while (!Thread.interrupted()) {
selector.select();
Set<SelectionKey> selected = selector.selectedKeys();
printSelectedKeys(selected);
Iterator<SelectionKey> it = selected.iterator();
while (it.hasNext()) {
dispatch(it.next());
}
selected.clear();
}
} catch (IOException e) {
e.printStackTrace();
}
} private void dispatch(SelectionKey key) {
Runnable r = (Runnable) key.attachment();
if (r != null) {
r.run();
}
} private void printSelectedKeys(Set<SelectionKey> selected) {
List<String> keyInfo = new ArrayList<>(selected.size());
for (SelectionKey sk : selected) {
String channelInfo = "";
if (sk.channel() instanceof SocketChannel) {
channelInfo = "SocketChannel port " + ((SocketChannel) sk.channel()).socket().getPort();
} else {
channelInfo = "ServerSocketChannel";
}
String readyOps = "";
if ((sk.readyOps() & SelectionKey.OP_ACCEPT) > 0) {
readyOps += "ACCEPT ";
}
if ((sk.readyOps() & SelectionKey.OP_CONNECT) > 0) {
readyOps += "CONN ";
}
if ((sk.readyOps() & SelectionKey.OP_READ) > 0) {
readyOps += "READ ";
}
if ((sk.readyOps() & SelectionKey.OP_WRITE) > 0) {
readyOps += "WRITE ";
}
String interestOps = "";
if ((sk.interestOps() & SelectionKey.OP_ACCEPT) > 0) {
interestOps += "ACCEPT ";
}
if ((sk.interestOps() & SelectionKey.OP_CONNECT) > 0) {
interestOps += "CONN ";
}
if ((sk.interestOps() & SelectionKey.OP_READ) > 0) {
interestOps += "READ ";
}
if ((sk.interestOps() & SelectionKey.OP_WRITE) > 0) {
interestOps += "WRITE ";
}
keyInfo.add(String.format("[%s, interestOps: %s, readyOps: %s]", channelInfo, interestOps, readyOps));
}
System.out.println(String.join(", ", keyInfo));
} private class Acceptor implements Runnable { @Override
public void run() {
try {
SocketChannel c = serverSocket.accept();
if (c != null) {
new Handler(selector, c);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

gordon.study.socket.nio.reactor.singlethread.Handler.java

public final class Handler implements Runnable {

    private final SocketChannel socket;

    private final SelectionKey sk;

    private ByteBuffer input = ByteBuffer.allocate(1024);

    private ByteBuffer output = ByteBuffer.allocate(1024);

    private static final int READING = 0, SENDING = 1;

    private int state = READING;

    public Handler(Selector sel, SocketChannel c) throws IOException {
socket = c;
c.configureBlocking(false);
sk = socket.register(sel, 0);
sk.attach(this);
sk.interestOps(SelectionKey.OP_READ);
} @Override
public void run() {
try {
if (state == READING) {
read();
} else if (state == SENDING) {
send();
}
} catch (IOException e) {
e.printStackTrace();
}
} private boolean inputIsComplete() {
if (input.position() >= 4) {
int length = input.getInt();
return input.position() >= length;
}
return false;
} private boolean outputIsComplete() {
return output.remaining() == 0;
} private void process() {
input.flip();
byte[] bytes = new byte[input.getInt() - 4];
input.get(bytes);
String msg = new String(bytes);
int remotePort = socket.socket().getPort();
System.out.println(remotePort + " Processing ... " + msg);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
}
System.out.println(remotePort + " Processed ... " + msg);
output.put((byte) 'Y');
output.flip();
} private void read() throws IOException {
socket.read(input);
if (inputIsComplete()) {
process();
state = SENDING;
sk.interestOps(SelectionKey.OP_WRITE);
}
} private void send() throws IOException {
socket.write(output);
if (outputIsComplete()) {
sk.cancel();
}
}
}

gordon.study.socket.nio.reactor.singlethread.Main.java

public class Main {

    public static void main(String[] args) throws IOException {
new Thread(new Reactor(8888)).start();
for (int i = 0; i < 10; i++) {
new Thread(new SocketClient()).start();
}
} private static class SocketClient implements Runnable { private String[] msgArray = { "ni hao", "hello", "chi le ma?", "你瞅啥?", "hi dude" }; @Override
public void run() {
try (Socket socket = new Socket()) {
socket.connect(new InetSocketAddress(8888));
System.out.printf(" client (port %d) connected to server.\n", socket.getLocalPort());
DataInputStream dis = new DataInputStream(socket.getInputStream());
DataOutputStream dos = new DataOutputStream(socket.getOutputStream());
int pos = ThreadLocalRandom.current().nextInt(msgArray.length);
sendMsg(msgArray[pos], dos);
char result = (char) dis.read();
System.out.printf(" client (port %d) get response from server: %s\n", socket.getLocalPort(), result);
dis.close();
dos.close();
} catch (Exception e) {
e.printStackTrace();
}
} private void sendMsg(String msg, DataOutputStream dos) throws Exception {
byte[] bytes = msg.getBytes();
int totalLength = 4 + bytes.length;
dos.writeInt(totalLength);
dos.write(bytes);
}
}
}

分析代码可以看出 Selector 是 Reactor 的核心,所有的 Channel 都会注册到 Selector 上。每个 Channel 对应一个 Handler 实例(Acceptor 视为特殊的 Handler),该 Handler 实例作为附件(attachment)附加在 SelectionKey 上。

这样,分发逻辑就十分简单了:当 Selector 选出 selectedKeys 时,遍历每个 key,拿出其附带的 Handler,执行其 run 方法即可。显然,Acceptor 与 Handler 实现 Runnable 接口的目的并不是为了多线程,只是为了有个共同的抽象(定义一个 AbstractHandler 替换 Runnable 的使用会更加容易理解一些)。

执行 main 方法,可以感受到单线程的效率低下(因为业务逻辑中 sleep 了一秒钟)。

有一个疑惑是关于 readyOps,从日志可以看出,当客户端发送了数据,SelectionKey 被选中时,readyOps 居然只包含 READ,而不包含 WRITE。在原来的理解中,这时 Channel 是可以向客户端发送数据的,所以 readyOps 应该包含 WRITE 才对啊?虽然可以写一段代码确认 readyOps 是不是受限于当前的 interestOps,但是意义不大,交给下一轮学习直接看源码吧。

B. 工作线程池版

一个最直接的优化思路就是将解码、业务处理和编码这些与 IO 无关的操作放到工作线程池中运行,以提高的 Reactor 的效率。

gordon.study.socket.nio.reactor.multithread.Handler.java

public final class Handler implements Runnable {

    private final SocketChannel socket;

    private final SelectionKey sk;

    private ByteBuffer input = ByteBuffer.allocate(1024);

    private ByteBuffer output = ByteBuffer.allocate(1024);

    private static Executor executor = Executors.newCachedThreadPool();

    private static final int READING = 0, SENDING = 1, PROCESSING = 2;

    private int state = READING;

    public Handler(Selector sel, SocketChannel c) throws IOException {
socket = c;
c.configureBlocking(false);
sk = socket.register(sel, 0);
sk.attach(this);
sk.interestOps(SelectionKey.OP_READ);
} @Override
public void run() {
try {
if (state == READING) {
read();
} else if (state == SENDING) {
send();
}
} catch (IOException e) {
e.printStackTrace();
}
} private boolean inputIsComplete() {
if (input.position() >= 4) {
int length = input.getInt();
return input.position() >= length;
}
return false;
} private boolean outputIsComplete() {
return output.remaining() == 0;
} private void process() {
input.flip();
byte[] bytes = new byte[input.getInt() - 4];
input.get(bytes);
String msg = new String(bytes);
int remotePort = socket.socket().getPort();
System.out.println(remotePort + " Processing ... " + msg);
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
}
System.out.println(remotePort + " Processed ... " + msg);
output.put((byte) 'Y');
output.flip();
} private void read() throws IOException {
socket.read(input);
if (inputIsComplete()) {
state = PROCESSING;
executor.execute(new Processer());
}
} private void send() throws IOException {
socket.write(output);
if (outputIsComplete()) {
sk.cancel();
}
} private void processAndHandOff() {
process();
state = SENDING;
sk.interestOps(SelectionKey.OP_WRITE);
sk.selector().wakeup(); // important!
} private class Processer implements Runnable {
public void run() {
processAndHandOff();
}
}
}

关于代码,在我看来,Handler 依然是线程安全的,所以没必要使用 synchronize 关键字。

第85行调用 selector 的 wakeup 方法很重要,否则服务端不会将响应内容发送给客户端。表面上的原因显然是第84行的 set interest ops 操作(发生在某个工作线程中)没有对 Reactor 所在线程(即主线程)当前阻塞的方法 Selector.select() 生效,所以通过 wakeup 方法强行终止掉本次阻塞,以期待下次 select 方法能接收到该 Channel 的 WRITE 事件,使 dispatch 循环正常运行下去。深层原因(为什么不生效)留给下一轮看源码分析吧。

C. 多 Reactor 版

如 06 篇所分析,处理连接建立的 Reactor(Selector)与处理数据传输的 Reactor(Selector)需要分开。更进一步,为了利用多核的能力,处理数据传输的 Reactor 应该有多个。

下图中 mainReactor 用于处理连接,subReactor 用于处理数据传输。

gordon.study.socket.nio.reactor.multireactor.Reactor.java

public class Reactor implements Runnable {

    private Selector selector;

    private ServerSocketChannel serverSocket;

    public Reactor(int port) throws IOException {
selector = Selector.open();
serverSocket = ServerSocketChannel.open();
serverSocket.socket().bind(new InetSocketAddress(port));
serverSocket.configureBlocking(false);
SelectionKey sk = serverSocket.register(selector, SelectionKey.OP_ACCEPT);
sk.attach(new Acceptor());
} public Reactor(Selector selector) throws IOException {
this.selector = selector;
} @Override
public void run() {
try {
while (!Thread.interrupted()) {
if(selector.select(100) == 0) {
continue;
}
Set<SelectionKey> selected = selector.selectedKeys();
printSelectedKeys(selected);
Iterator<SelectionKey> it = selected.iterator();
while (it.hasNext()) {
dispatch(it.next());
}
selected.clear();
}
} catch (IOException e) {
e.printStackTrace();
}
} private void dispatch(SelectionKey key) {
Runnable r = (Runnable) key.attachment();
if (r != null) {
r.run();
}
} private void printSelectedKeys(Set<SelectionKey> selected) {
} private class Acceptor implements Runnable { Selector[] selectors = new Selector[2]; int next = 0; public Acceptor() throws IOException {
Executor executor = Executors.newFixedThreadPool(selectors.length);
for (int i = 0; i < selectors.length; i++) {
selectors[i] = Selector.open();
Reactor subReactor = new Reactor(selectors[i]);
executor.execute(subReactor);
}
} @Override
public void run() {
try {
SocketChannel c = serverSocket.accept();
System.out.printf(" server established connection: %d\n", c.socket().getPort());
if (c != null) {
new Handler(selectors[next], c);
}
if (++next == selectors.length) {
next = 0;
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
}

我选择让 Acceptor 创建 subReactor(s),并通过 Round Robin 的方式将连接平均分配给 subReactors。 每个 subReactor 都运行在一个独立线程中。

代码24行选择了带超时的 select 方法,否则无论我怎样调用 wakeup,程序总是有概率卡死,原因不明。

Java网络编程学习A轮_08_NIO的Reactor模型的更多相关文章

  1. Java网络编程学习A轮_01_目标与基础复习

    A. A轮目标 复习网络编程基础知识,重点学习下TCP三次握手四次挥手,以及可能引发的异常情况. 回顾 Socket 编程,好多年没写(chao)过相关代码了. 重学 NIO,以前学的基本忘光了,毕竟 ...

  2. Java网络编程学习A轮_06_NIO入门

    参考资料: 老外写的教程,很适合入门:http://tutorials.jenkov.com/java-nio/index.html 上面教程的译文:http://ifeve.com/overview ...

  3. Java网络编程学习A轮_07_基于Buffer的Socket编程

    示例代码: https://github.com/gordonklg/study,socket module A. LineSeparate 基于 Buffer 实现逐行读取的 EchoServer ...

  4. Java网络编程学习A轮_05_Socket编程

    示例代码: https://github.com/gordonklg/study,socket module A. Socket 编程简单例子 最简单的 Socket 编程是通过回车/换行符,整行读取 ...

  5. Java网络编程学习A轮_03_抓包分析TCP四次挥手

    参考资料: http://www.jellythink.com/archives/705 示例代码: https://github.com/gordonklg/study,socket module ...

  6. Java网络编程学习A轮_04_TCP连接异常

    参考资料: https://huoding.com/2016/01/19/488 示例代码: https://github.com/gordonklg/study,socket module A. C ...

  7. Java网络编程学习A轮_02_抓包分析TCP三次握手过程

    参考资料: https://huoding.com/2013/11/21/299 https://hpbn.co/building-blocks-of-tcp/#three-way-handshake ...

  8. Java 网络编程学习总结

    新手一枚,Java学习中,把自己学习网络编程的知识总结一下,梳理下知识,方便日后查阅,高手莫进. 本文的主要内容: [1]    网络编程认识                [2]  TCP/IP编程 ...

  9. Java网络编程学习笔记

    Java网络编程,我们先来看下面这一张图: 由图可得:想要进行网络编程,首先是服务器端通过ServerSocket对某一个端口进行监听.通过accept来判断是否有客户端与其相连.若成功连上,则通过r ...

随机推荐

  1. org.apache.log4j日志级别

    日志记录器(Logger)是日志处理的核心组件.log4j具有7种级别(Level).日志记录器(Logger)的可用级别Level (不包括自定义级别 Level)优先级从高到低:OFF.FATAL ...

  2. Spring源码学习之IOC实现原理(二)-ApplicationContext

    一.Spring核心组件结构 总的来说Spring共有三个核心组件,分别为Core,Context,Bean.三大核心组件的协同工作主要表现在 :Bean是包装我们应用程序自定义对象Object的,O ...

  3. 信息收集1:DNSEUM命令

    1,背景 今天无意中发现了dnsenum这个工具,在网上搜了下关于dnsenum的介绍和安装使用方法,资料不是很全,但还好这个工具也算简单,网上也都有源码,可以自行下载下来阅读阅读.本人好奇在本机(u ...

  4. nginx 上php不可写解决方法

    在php.ini中设置的session.save_path会被php-fpm.conf中覆盖 打开php-fpm.conf文件找到php_value['session.save_apth'] 这里的/ ...

  5. talib 中文文档(八): Momentum Indicator Functions 动量指标

    Momentum Indicator Functions ADX - Average Directional Movement Index 函数名:ADX 名称:平均趋向指数 简介:使用ADX指标,指 ...

  6. 106 miles to Chicago---zoj2797(最短路问题,求概率,模板)

    题目链接:http://www.icpc.moe/onlinejudge/showProblem.do?problemId=1797 题意是有 n 个点 m 条边,从a到b的不被抓的概率是p,让求从点 ...

  7. 【我的Android进阶之旅】解决sqlcipher库:java.lang.IllegalStateException: get field slot from row 0 col 0 failed.

    一.背景 最近维护公司的大数据SDK,在大数据SDK里面加入了ANR的监控功能,并将ANR的相关信息通过大数据埋点的方式记录到了数据库中,然后大数据上报的时候上报到大数据平台,这样就可以实现ANR性能 ...

  8. 怎么应对 domino文档损坏然后损坏文档别删除导致数据丢失

    对于domino 有个机制是同步 ..然后如果文档被损坏之后会通过同步或者压缩 之类的 然后将损坏文档删除 那么这样就有个风险..知识管理文档会被删除. 并且删除了之后管理员如果不仔细看日志的话也不会 ...

  9. (转)JSON Web Token - 在Web应用间安全地传递信息

    JSON Web Token(JWT)是一个非常轻巧的规范.这个规范允许我们使用JWT在用户和服务器之间传递安全可靠的信息. 让我们来假想一下一个场景.在A用户关注了B用户的时候,系统发邮件给B用户, ...

  10. MISC-WHCTF2016-crypto100

    题目:李二狗的梦中情人  找不同! 如图,下载得到“nvshen.png” 流程:看到这个被命名为nvshen的文件,感觉文件本身会有东西.用16进制查看器在图片的末尾发现了一串类似URL的ASCII ...