`
pelli
  • 浏览: 6196 次
  • 性别: Icon_minigender_1
  • 来自: 上海
社区版块
存档分类
最新评论

tcp nio basics

    博客分类:
  • Java
阅读更多
nio(non-blocking io)是由操作系统实际执行阻塞的网络io:应用将要发送的数据写到某个缓冲区,由操作系统实际发送出去;操作系统接收到数据后放到某个缓冲区,供应用直接读取。由于阻塞操作都交给了操作系统,所以应用通常不会阻塞。nio不同于asynchronous io,后者是将阻塞的io操作交由应用内的某线程池去执行。

nio编程围绕着java.nio.channels.Selector:有连接过来了、连上了对方、有数据可读、有数据可写 - 所有期待被通知这四种事件的Socket/ServerSocket都注册到Selector,当事件发生时应用会知道并做相应处理。以下是一个简单例子NServer(接收input并响应hello input)。

首先创建一个Selector:
Selector selector = Selector.open();


创建一个Server socket在8888端口监听:
ServerSocketChannel serverCh = ServerSocketChannel.open();
serverCh.bind(new InetSocketAddress(8888));
serverCh.configureBlocking(false);
serverCh.register(selector, SelectionKey.OP_ACCEPT);

在注册到selector之前要设置server socket为non-blocking的。注册时指定此server socket对OP_ACCEPT事件感兴趣。

以下是一个常见的无限循环,每次循环调用selector.select()方法得到所有感兴趣的事件发生了的Socket/ServerSocket,然后逐一处理。下面是示意结构代码(具体代码参考附件):
for(;;) {
    selector.select();
    Iterator<SelectionKey> ite = selector.selectedKeys().iterator();
    for (; ite.hasNext() ;) {
        SelectionKey key = ite.next();
        if(key.isAcceptable()) {
            //有连接过来了

            ite.remove();
        } else if(key.isReadable()) {
            //有数据可读

            ite.remove();
        } else if(key.isWritable()) {
            //有数据可写

            ite.remove();
        }
    }
}

上面注意每次处理完后要从iteration里移除当前key(不移除会一直在)。

SelectionKey代表在Selector上的注册,SelectionKey.channel()方法返回注册的Socket/ServerSocket。OP_ACCEPT事件处理如下:
if(key.isAcceptable()) {
    SocketChannel ch = ((ServerSocketChannel) key.channel()).accept();
    ch.configureBlocking(false);
    SelectionKey key2 = ch.register(selector, SelectionKey.OP_READ);
    key2.attach(new Attach("Ch-" + ++chCount));

    ite.remove();
}

OP_ACCEPT事件一定发生在前面在8888端口监听的ServerSocket上,调用它的accept()方法直接获取到(已经由操作系统建立好的)连接socket。我们这里演示socket接收到input会响应hello input,所以马上把该socket注册到selector 并指定对OP_READ感兴趣。
SelectionKey允许attach任意对象。由于程序中会有很多socket并且每个socket发送的数据都不同,所以定制了一个特定的Attach类用来存储socket的名称和将发送的数据:
static class Attach {
    String name;
    String output;

    Attach(String name) {
        this.name = name;
    }

    static String nameOf(SelectionKey key) {
        Attach a = (Attach) key.attachment();
        return a.name;
    }

    static void output(SelectionKey key, String output) {
        Attach a = (Attach) key.attachment();
        a.output = output;
    }

    static String outputOf(SelectionKey key) {
        Attach a = (Attach) key.attachment();
        return a.output;
    }
}


当socket的客户端发来数据时,操作系统会接收到然后程序会进入OP_READ分支:
else if(key.isReadable()) {
    SocketChannel ch = (SocketChannel) key.channel();
    buf.position(0);
    int len;
    try {len = ch.read(buf); }
    catch (IOException e) {
        len = -1;
    }
    if(len == -1) {
        System.out.println(Attach.nameOf(key) + " Read: EOF");
        key.interestOps(key.interestOps() & ~SelectionKey.OP_READ);
    } else if(len > 0) {
        String input = new String(buf.array(), 0, len).trim(); // trailing \r\n
        System.out.println(Attach.nameOf(key) + " Read: " + input);

        if("close".equals(input)) {
            System.out.println(Attach.nameOf(key) + " Close");
            ch.close();
        }
        else if("exit".equals(input)) {
            System.out.println(Attach.nameOf(key) + " Exit");
            ch.close();
            exit = true;
            break;
        }
        else {
            System.out.println(Attach.nameOf(key) + " will Write: " + input);
            key.interestOps(key.interestOps() | SelectionKey.OP_WRITE);
            Attach.output(key, "Hello " + input);
        }
    } else if(len == 0) {
        System.out.println(Attach.nameOf(key) + " Read: len=0");
    }

    ite.remove();
}

因为是isReadable,读数据是不会阻塞的,但注意SocketChannel.read()方法可能返回-1或者抛出异常。经实验:
a)直接强关telnet客户端 会返回-1,无异常。
b)直接强关NClient(例子程序,见后) 会异常“远程主机强迫关闭了一个现有的连接”
所以凡是io读、写 都可能发生IOException,程序需要自行处理。

当-1代表此socket已不可读(每个socket都有互相独立的InputStream、OutputStream 2个流),程序取消对OP_READ的兴趣(需要关闭socket?)。如果读到实际的内容input(排除"close"和"exit")要准备响应hello input。由于是nio 需要等到操作系统提示可写时才能写,所以程序将待写内容存入Attach 并增加对OP_WRITE的兴趣。另,len=0这个分支在测试中没有遇到过。

下一步当提示可写时,程序从Attach取出数据、写出、然后取消OP_WRITE兴趣:
else if(key.isWritable()) {
    SocketChannel ch = (SocketChannel) key.channel();
    String output = Attach.outputOf(key);
    System.out.println(Attach.nameOf(key) + " Write: " + output);
    ch.write(ByteBuffer.wrap((output + "\n").getBytes()));
    key.interestOps(key.interestOps() & ~SelectionKey.OP_WRITE);

    ite.remove();
}

注意为了简化,示例程序调用SocketChannel.write()并没有检查内容是否全部写出。按照API说明,write方法甚至可能返回0 - 代表一个字节也没写出(到操作系统的某个缓存区)。

如上是一个简单的nio Server例子。运行该程序,然后启动一个telnet客户端“telnet localhost 8888”,输入"abc",看到程序输出:
Ch-1 Read: abc
Ch-1 will Write: abc
Ch-1 Write: Hello abc

输入"close"关闭当前连接,程序输出:
Ch-1 Read: close
Ch-1 Close

再次运行telnet,输入"def":
Ch-2 Read: def
Ch-2 will Write: def
Ch-2 Write: Hello def

再运行一个telnet,输入"xyz":
Ch-3 Read: xyz
Ch-3 will Write: xyz
Ch-3 Write: Hello xyz

输入"exit"结束。


测试客户端NClient

同样,建立一个唯一的Selector,然后注册任意多个socket,指定对OP_CONNECT和OP_READ感兴趣:
Selector selector = Selector.open();

for(int i=0; i<1; i++) {
    SocketChannel ch = SocketChannel.open();
    ch.configureBlocking(false);
    SelectionKey key = ch.register(selector, SelectionKey.OP_CONNECT | SelectionKey.OP_READ);
    key.attach(new Attach("Ch-" + (i+1)));
    ch.connect(new InetSocketAddress(8888));
}


nio的客户端当到远端的连接就绪时:
if (key.isConnectable()) {
    SocketChannel ch2 = (SocketChannel) key.channel();
    ch2.finishConnect();
    key.interestOps(key.interestOps() & ~SelectionKey.OP_CONNECT);

    initWrite(key);

    ite.remove();
}

通过SocketChannel.finishConnect()完成客户端的连接。连上后客户端准备主动发数据:
static void initWrite(SelectionKey key) {
    String output = Attach.nameOf(key) + " " + Math.random();
    System.out.println(Attach.nameOf(key) + " will Write: " + output);
    key.interestOps(key.interestOps() | SelectionKey.OP_WRITE);
    Attach.output(key, output);
}


当操作系统提示可写时把数据写出,并取消OP_WRITE兴趣:
else if(key.isWritable()) {
    SocketChannel ch2 = (SocketChannel) key.channel();
    String output = Attach.outputOf(key);
    System.out.println(Attach.nameOf(key) + " Write: " + output);
    ch2.write(ByteBuffer.wrap((output + "\n").getBytes()));
    key.interestOps(key.interestOps() & ~SelectionKey.OP_WRITE);

    ite.remove();
}

同样注意为了简化这里没有检查SocketChannel.write()的返回值。

当NServer 收到input、响应数据过来时,发送下一条数据:
else if(key.isReadable()) {
    SocketChannel ch2 = (SocketChannel) key.channel();
    buf.position(0);
    int len = ch2.read(buf);
    if(len > 0) {
        String input = new String(buf.array(), 0, len).trim(); // trailing \n
        System.out.println(Attach.nameOf(key) + " Read: " + input);

        initWrite(key);

        ite.remove();
    }
}


留意一下if isWritable和else if isReadable是分支处理,但nio 实际上(估计)应该存在同时可写、可读的情况。


将socket数量改到1000,运行NServer、NClient,CPU迅速升到90%以上(NClient里每次循环sleep了10毫秒)。通过telnet 输入"exit"结束。简单实验可以看到NServer 一根线程可以处理1000个并发连接的连续读写,nio相对传统blocking io的优势就是在对线程的节省上。


(测试环境:Windows10+Java8、CentOS7+Java8)
分享到:
评论
发表评论

文章已被作者锁定,不允许评论。

相关推荐

    分布式Java中的TCP/IP+NIO

    在分布式Java应用中,TCP/IP协议和NIO(非阻塞I/O)是构建高性能、高可用性系统的关键技术。TCP/IP是一种传输层协议,确保数据在网络中的可靠传输,而NIO是Java提供的一个I/O模型,允许程序进行非阻塞的数据读写,...

    tcp.zip_java Tcp _java nio_java nio TCP_tcp_tcp java

    在这个“tcp.zip”压缩包中,我们可能找到了关于使用Java实现TCP服务器和利用NIO处理多个客户端连接的相关资料。 TCP(传输控制协议)是互联网协议栈中的核心部分,它保证了数据的可靠传输。在Java中,我们可以使用...

    默蓝网络通信测试工具(NIOSocket工具)支持TCP/IP和HTTP通信-网络通信开发人员必备

    "默蓝网络通信测试工具(NIOSocket工具)"正是这样一款针对TCP/IP和HTTP通信的专业测试工具,它为网络通信开发人员提供了强大的测试与分析能力。 首先,我们要理解TCP/IP协议的重要性。TCP(传输控制协议)和IP(网际...

    java_Netty权威指南,详解nio,tcp,http,netty

    这个权威指南深入探讨了Netty的核心概念和技术,同时也涵盖了与网络编程相关的基础,如非阻塞I/O(NIO)、传输控制协议(TCP)以及超文本传输协议(HTTP)。下面将对这些知识点进行详细解释。 一、非阻塞I/O(NIO)...

    BIO、NIO、AIO、Netty 、TCP全网最全解析!Netty中提供了哪些线程模型?

    本文将深入探讨BIO( Blocking I/O)、NIO(Non-blocking I/O)、AIO(Asynchronous I/O)以及Netty框架中的线程模型,并与TCP网络协议相结合,为您提供全网最全面的解析。 首先,让我们从基础开始,了解这些I/O...

    nio学习demo 处理因缓冲区满导致写入失败问题

    总的来说,这个示例通过NIO实现了客户端与服务器之间的高效通信,解决了TCP沾包问题,同时确保了在缓冲区满时能优雅地处理写入失败,避免阻塞。在实际应用中,可以根据需求调整缓冲区大小、优化解码策略,以及改进...

    NIO 入门.chm,NIO 入门.chm

    6. **网络I/O**:NIO为TCP和UDP提供了SocketChannel和DatagramChannel,用于处理网络连接。SocketChannel用于TCP连接,而DatagramChannel则用于UDP的无连接通信。 7. **通道与缓冲区的交互**:数据总是通过通道读入...

    Java NIO英文高清原版

    5. **网络通道(Network Channels)**:如SocketChannel和ServerSocketChannel,它们用于处理TCP和UDP网络通信。SocketChannel用于客户端连接,ServerSocketChannel用于监听并接受新的连接请求。 6. **多路复用器...

    java NIO.zip

    Java NIO,全称为Non-Blocking Input/Output(非阻塞输入/输出),是Java标准库提供的一种替代传统的I/O模型的新技术。自Java 1.4版本引入NIO后,它为Java开发者提供了更高效的数据传输方式,尤其是在处理大量并发...

    JavaNIO chm帮助文档

    Java NIO系列教程(一) Java NIO 概述 Java NIO系列教程(二) Channel Java NIO系列教程(三) Buffer Java NIO系列教程(四) Scatter/Gather Java NIO系列教程(五) 通道之间的数据传输 Java NIO系列教程(六)...

    Java Socket编程实例(四)- NIO TCP实践

    本实例主要关注NIO在TCP中的应用,它允许更高效的资源管理和处理多个连接,特别适合高并发场景。我们将探讨以下几个关键知识点: 1. **NIO(Non-blocking Input/Output)**: 与传统的BIO不同,NIO是非阻塞的,这...

    Java NIO系列教程(一) Java NIO 概述

    ### Java NIO 系列教程(一):Java NIO 概述 #### 一、引言 Java NIO(New IO)是Java SE 1.4版本引入的一个新的I/O处理框架,它提供了比传统Java IO包更高效的数据处理方式。NIO的核心在于其三大组件:Channels...

    用NIO解析modbus协议

    标题中的“用NIO解析modbus协议”表明我们要探讨的是如何使用Java的非阻塞I/O(Non-blocking Input/Output,简称NIO)框架来处理Modbus通信协议。Modbus是一种广泛使用的工业通信协议,主要在自动化设备之间进行数据...

    java NIO实例

    5. **SocketChannel**:用于网络通信,可以建立TCP连接,进行非阻塞的读写。在`NIOServer.java`中,可能会创建ServerSocketChannel监听客户端连接,然后通过accept()方法接收新连接,形成SocketChannel。 6. **Pipe...

    java nio 尚硅谷 12讲 new

    - SocketChannel和ServerSocketChannel分别用于处理TCP连接和监听。DatagramChannel则用于UDP协议的无连接通信。 7. **多路复用(Multiplexing)** - 通过选择器,一个线程可以同时处理多个通道的事件,实现高效...

    使用NIO方式完成简单的通信

    在Java编程中,NIO(New IO)是一种与传统IO不同的I/O模型,它提供了非阻塞的输入/输出操作,适用于高并发、大数据量的网络应用。本教程将介绍如何使用NIO来实现简单的通信,包括消息头和消息体的设计。 首先,我们...

    NIO入门pdf分享

    6. **网络通信**:NIO为网络通信提供了SocketChannel和ServerSocketChannel,它们可以实现TCP和UDP的非阻塞连接。 7. **scatter/gather传输**:NIO支持scatter/gather传输,即数据可以从多个缓冲区写入通道,也可以...

    《NIO与Socket编程技术指南》_高洪岩

    Socket通信基于TCP/IP协议,确保数据的可靠传输,通过输入流和输出流进行数据交换。在实际应用中,Socket常用于实现分布式服务、聊天应用、文件传输等场景。 本书可能涵盖了以下主题: 1. NIO基础:介绍NIO的基本...

Global site tag (gtag.js) - Google Analytics