用Java来实现BIO和NIO模型的HTTP服务器(二) NIO的实现

前言

这是一个系列的文章,按照规划是用Java标准库、Netty来实现一个非常简单的HTTP服务器,HTTP服务器我们可以使用Java标准库提供的api,实现BIO、NIO模型的HTTP服务器,然后再用Netty实现,前一篇我们写的类在这一篇还可以用到,让我们回忆一下上一篇我们讲了什么,我们回顾了通信的发展史,从最开始的点对点链路,到总线链路,再到mac地址,ip地址,最后引出两台计算机之间的通信事实上是两台计算机上面进程之间的通信,那么该数据包到达计算机之后该如何交给哪个进程呢,这也就是端口,运输层引入了端口的概念,ip+端口构成TCP连接的一端,那么要通信就首先要建立连接,也就是三次握手,连接建立之后就可以通过连接来传输数据了,那么该如何管理连接呢? 操作系统在连接建立的时候,会将这个消息通知给进程。

我们的程序通过Socket来和操作系统进行交互,这里的Socket指的是操作系统提供的服务,当一个进程向一个进程发起请求建立连接的请求,这个数据包首先经过操作系统提供的接口向下传递,然后通过互联网中层层设备转发来到另一个进程所在的计算机上,两台计算机完成连接建立之后,通知上层的应用程序。

当我们编写的应用程序需要使用网络服务的时候,在Java中我们首先要明确自己是客户端还是服务端,客户端是发起请求的一方, 我们客户端的代码可以这么写:

java 复制代码
Socket socket = new Socket();
// 代表客户端请求连接ip地址为127.0.0.1,端口为8080的进程
socket.connect(new InetSocketAddress("127.0.0.1",8080));
// 连接之后获取输入流
OutputStream outputStream = socket.getOutputStream();
// 写入hello world
outputStream.write("hello world".getBytes());

客户端在发起连接请求的时候,这个请求会首先到达操作系统,操作系统会为这次调用所需要的一些资源(CPU时间,网络带宽、存储器空间等)分配该应用进程。操作系统为这些资源总和创建一个套接字描述符的好嘛来表示,然后将这个套接字描述返回给应用进程。看到这里有一个疑问,客户端没声明自己在那个端口上,那服务端在给客户端发送消息的时候,这个消息到达操作系统应该给谁呢? 答案是操作系统会从可用的端口分配一个,但是如果你想绑定指定的端口其实也可以,Socket这个类里面提供了bind方法:

java 复制代码
public void bind(SocketAddress bindpoint) throws IOException 

客户端写完之后我们来写服务端, TCP协议中我们需要一个服务端,监听指定的端口。在Java里面写服务端的应用程序事实上有两套API,一套是JDK 1.0引入的以ServerSocket为中心的API,一套是JDK 1.4 引入的以ServerSocketChannel为核心的API。第一套写监听的方法如下:

ini 复制代码
ServerSocket serverSocket = new ServerSocket();
// 绑定在8080端口
serverSocket.bind(new InetSocketAddress(8080));
// 监听连接,该方法会阻塞到这里直到有连接建立完成
// 发起系统调用
Socket socket = serverSocket.accept();
while (true){
    InputStream socketInputStream = socket.getInputStream();
    byte[] readByte = new byte[4096];
    // 这里其实数据不见得立马可以读, 因为数据不代表立马可以读
    // 发起系统调用
    int readTotalNumber  = socketInputStream.read(readByte);
    String s = new String(readByte,0,readTotalNumber);
    System.out.println(s);
}

这里我们详细的解释一下为什么数据为什么不可读,我们知道现代高级语言程序想要使用网络服务,必须调用操作系统提供的接口,这种调用也被称为系统调用,发生系统调用的时候发生了什么?

系统调用指运行在用户空间(User space)向操作系统内核请求需要更高权限运行的服务。

让我们回忆一下操作系统的内核空间和用户空间,计算机的内存被切割为两个部分:

用户空间: 正如同的它的名字一样,处内核以外所有的用户进程运行在这个空间上。内核的作用是管理在该空间内运行的应用程序,防止他们互相干扰,避免机器出现混乱。

内核空间: 内核的代码和数据存放在这个位置上,内核也是一个进程,内核运行在这块内存之上。

与之相对的两个概念是内核模式(Kernel mode,有资料也称为System Model 系统模式),是Linux中CPU运行模式之一。另一种是用户模式(user model),是用户程序的非特权模式,也就是内核以外的所有操作模式。当CPU运行在内核模式下面,默认运行的是受信任的程序,因此它可以执行任何指令和访问任何内存位置。内核(操作系统的核心,对系统中发生的一切拥有完全的控制权)是被信任的软件,其他程序不受信任,因此所用的用户进程都必须使用系统调用来请求内核执行特权指令,比如创建进程、I/O操作。

术语System Call 和 System Exit是实际汇编语言指令的占位符,分别用于将CPU从用户模式切换到内核模式,从内核模式切换到用户模式。当用户进程发起一个调用,Linux会为这个调用分配一个系统调用编号,Linux使用系统调用表(System Call Dispatch Table)存储调用编号和实际执行系统调用对应功能的函数。

实际运行上面的程序会发现 , 会出现下面的异常:

php 复制代码
Exception in thread "main" java.net.SocketException: Connection reset
    at java.net.SocketInputStream.read(SocketInputStream.java:210)
    at java.net.SocketInputStream.read(SocketInputStream.java:141)
    at java.net.SocketInputStream.read(SocketInputStream.java:127)
    at com.example.quicktest.ServerSocketDemo.oldAPI(ServerSocketDemo.java:42)
    at com.example.quicktest.ServerSocketDemo.main(ServerSocketDemo.java:20)

原因在于我们上面写的程序是是在不断的处理连接的,收到数据之后,再读收到了RST包,那什么是RST包,让我们回忆一下关于TCP的经典面试题三次握手和四次挥手:

声明图片来自于参考文档[16]

我们一边写这个一边将我们的所学联系起来,在TCP协议中客户端主动关闭连接,发起系统调用之后,内核发一个TCP数据包,这个TCP数据包的终止位FIN置成1,序号seq = u,它等于前面已传送的数据的最后一个字节的序号加1,这时Client进入FIN-WAIT-1(终止等待1)状态,等待服务端的确认。

服务端收到FIN之后,向Client发送确认包ack = u +1,这个u等于前面已经传送的数据的最后一个序号加1 , Client收到ack之后进入到FIN-WAIT-2,服务端如果没有数据要发送了就会向客户端发送TCP数据包,数据包中的FIN置为1, 服务端还必须重复上次已发送过的确认好ack = u + 1.这时B就进入LAST-ACK(最后确认)状态,等待A的确认。

客户端再次收到服务端的连接释放报文段后,必须对此发出确认。在确认报文段中把ACK置为1、确认号ack = w + 1 ,而自己的序号是seq = u + 1(根据TCP标准, 前面发送给的FIN报文段要消耗一个序号)。然后进入到TIME-WAIT(时间等待)的状态。请注意,现在TCP连接还没有释放掉。必须经过时间等待计时器(TIME-WAIT timer) 设置的时间2MSL后,A才进入到CLOSED状态

然后我们客户端退出之后,没有显式的调用close,也就是客户端没有走正常流程关闭TCP连接,但对于操作系统来说还是要回收对应的资源,所以进程退出的时候,内核会监测到这个变化,因为这个连接已经是异常了。 在传输控制协议(TCP)连接的数据包流中,每个数据包都包含一个TCP包头。这些包头中的每一个都包含一个称为"复位"(RST)标志的位。在大多数数据包中,该位设置为0,并且无效;但是,如果此位设置为1,则向接收计算机指示该计算机应立即停止使用TCP连接;它不应使用连接的标识号(端口)发送更多数据包,并丢弃接收到的带有包头的其他数据包,这些包头指示它们属于该连接。

所以上面的代码客户端完善一点应当是这个样子:

java 复制代码
// 借助try-with-resources 自动关闭释放资源
try (Socket socket = new Socket()){
    socket.connect(new InetSocketAddress("127.0.0.1",8080));
    try(OutputStream outputStream = socket.getOutputStream()){
        outputStream.write("hello world".getBytes());
    }
}

accept调用也应该放在whie(true)循环里面,所以代码应当改成下面这个样子:

ini 复制代码
ServerSocket serverSocket = new ServerSocket();
// 绑定在8080端口
serverSocket.bind(new InetSocketAddress(8080));
// 监听连接,该方法会阻塞到这里直到有连接建立完成
// 发起系统调用
while (true) {
    try (Socket socket = serverSocket.accept()) {
        try (InputStream socketInputStream = socket.getInputStream()) {
            byte[] readByte = new byte[4096];
            // 这里其实数据不见得立马可以读, 因为数据不代表立马可以读
            // 发起系统调用
            int readTotalNumber = socketInputStream.read(readByte);
            String s = new String(readByte, 0, readTotalNumber);
            System.out.println(s);
        }
    }
}

在这种模型下同时只能处理一下一个连接,因为我们只有一个线程,这个链接的读取逻辑没处理完毕,下一个得等待在那里。我们这里解释一下,我们调用SocketInputStream的read函数的时候为什么不是立刻能读取,一般来说,一台能联网的计算机首先得有网卡不管是有线网卡还是无线网卡,数据经过路由器之后网线到达网卡,然后将数据包从网卡硬件缓存转移到内存中,然后通知内核处理,然后经过TCP/IP协议层处理,最后应用程序通过系统调用读取到发送过来的数据。

上面的写法还面临的一个问题就是没有判断什么时候报文结束,如果是短链接即传输一次消息连接就关闭,那么read函数返回-1就代表数据结束,如果我们希望TCP连接保活,即保持这个链接,我们只是做示例,完整的会在下面构建HTTP服务器中详细讲述。

如果你熟悉网络编程,还有一个网络异常会经常碰到:

arduino 复制代码
Exception in thread "main" java.net.SocketException: Connection reset by peer

被对等方重置连接,这是啥意思? 相当于突然挂电话,这比单纯的不回应、让人等着更有礼貌。但这并不是真正有礼貌的TCP/IP的关闭方式。也就是说连接建立了,某一方突然关闭连接,另一方还在使用这个连接,就会出现这个异常。

在《用Java的BIO和NIO、Netty来实现HTTP服务器(一) 》里面我们用的是在1.4引入的新API,这套API的优势就是比较统一,可以通过ServerSockeChannel的configureBlocking来制定使用BIO还是NIO,所以上面服务端的写法可以等价转换为:

ini 复制代码
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.bind(new InetSocketAddress(8080));
while (true) {
    try (SocketChannel socketChannel = serverSocketChannel.accept()) {
        ByteBuffer byteBuffer = ByteBuffer.allocate(4096);
        socketChannel.read(byteBuffer);
        byteBuffer.flip();
        byte[] readDataArray = new byte[byteBuffer.limit()];
        byteBuffer.get(readDataArray);
        String readData = new String(readDataArray);
        System.out.println(readData);
    }
}

ByteBuffer基于byte数组封装了一些常见的操作,可以理解为一个容器,put存,get取,但是在取之前我们需要知道取到哪个位置,调用flip方法之后,调用limit方法之后就能知道ByteBuffer中放了多少元素。

前面面临的问题

(一)是七月份写的,这里我们在复习一下里面的内容, 我们希望构建用Java标准库,以及Java的NIO框架Netty实现一个简单的HTTP服务器,(一)是BIO模型,不管是NIO还是BIO,还是用框架构建,基于TCP协议的网络编程都会面临这样一个问题,首先是管理连接,也就是连接建立,连接建立之后我们读取数据,那么HTTP报文不固定大小,我们需要根据报文结束标志来判断是否读取结束,然后读取完整之后交给下一层去处理。但是NIO还是BIO他们具备共性,所以我们用继承来实现,我们首先抽象了一个Server的基础类:

java 复制代码
public abstract class Server {
​
    protected ServerSocketChannel serverSocketChannel;
​
    protected SSLContext sslContext;
​
    static int port = 8000;
​
    static int backlog = 1024;
​
    static boolean secure = false;
​
    public Server(int port, int backlog, boolean secure) throws IOException {
        // 创建一个ServerSocketChannel
        serverSocketChannel =  ServerSocketChannel.open();
        /**
         * 可以重用ip+端口这个链接,TCP以链接为单位,当TCP链接要关闭的时候,会等待一段时间再进行关闭,
         * 如果我想要重用端口,那么channel就无法绑定,在绑定到对应地址之前,设定重用地址。即使在这个端口上的tcp连接
         * 处于处于TIME_WAIT状态,我们仍然可以使用
         */
        serverSocketChannel.socket().setReuseAddress(true);
        // 绑定端口和backlog
        serverSocketChannel.bind(new InetSocketAddress(port),backlog);
​
    }
​
    /**
     * 这里是抽象方法,我们后面要用NIO再实现一遍
     * 所以这里交给子类来实现
     */
    protected  abstract void runServer();
​
    /**
     * 我们要写的是一个简单的HTTP服务器,
     * 这个服务器可以从命令行方式启动的时候接收参数
     * 我们可以选择从main函数
     * @param args
     */
    public static void main(String[] args) throws IOException {
        Server server = null;
        if (args.length == 0){
            System.out.println("http server running default model");
            server = new BlockingServer(port,backlog,secure);
            server.runServer();
        }
        // 端口目前先固定死, 我们目前只读一个参数
        if ("B".equals(args[0])){
            server = new BlockingServer(port,backlog,secure);
        }else if ("N".equals(args[0])){
            server = new NonBlockingServer(port,backlog,secure);
        }else{
            System.out.println("input args error only support B OR N");
            return;
        }
        server.runServer();
    }
}
​
public class BlockingServer extends Server {
    
    public BlockingServer(int port, int backlog, boolean secure) throws IOException {
        super(port, backlog, secure);
    }
    
    @Override
    protected void runServer() throws IOException {
        for (;;){
            SocketChannel socketChannel = serverSocketChannel.accept();
            ChannelIO channelIO = ChannelIO.getInstance(socketChannel, true);
            RequestServicer requestServicer = new RequestServicer(channelIO);
            requestServicer.run();
        }
    }
}

我们用ServerSocketChannel这个为核心来构建HTTP服务器,原因是实现上更为统一,我们最终可以做成一个jar,所以我们根据命令行参数来决定是BIO还是NIO模型的服务器。启动之后应当是一个无限循环,不断接收连接,不断处理请求。连接建立之后,我们需要不断的读数据,这是NIO和BIO共同的特征,所以我们写了一个ChannelIO工具类,来实现对数据的读取:

java 复制代码
public class ChannelIO {
​
    private SocketChannel socketChannel;
​
    private ByteBuffer requestBuffer;
​
    int defaultByteBufferSize = 4096;
​
    private ChannelIO(SocketChannel socketChannel, boolean blocking) throws IOException {
        this.socketChannel = socketChannel;
        this.requestBuffer = ByteBuffer.allocate(4096);
        this.socketChannel.configureBlocking(blocking);
    }
​
​
    public static ChannelIO getInstance(SocketChannel socketChannel, boolean blocking) throws IOException {
        return  new ChannelIO(socketChannel,blocking);
    }
​
    public int read() throws IOException {
        // 剩余的小于百分之五自动扩容
        resizeByteBuffer(defaultByteBufferSize / 20);
        return socketChannel.read(requestBuffer);
    }
​
    private void resizeByteBuffer(int remaining) {
        if (requestBuffer.remaining() < remaining){
            // 扩容一倍
            ByteBuffer newRequestBuffer = ByteBuffer.allocate(requestBuffer.capacity() * 2);
            // 转为读模式
            requestBuffer.flip();
            //  将旧的buffer放入到新的buffer中
            newRequestBuffer.put(requestBuffer);
            requestBuffer = newRequestBuffer;
        }
    }
​
    public ByteBuffer getReadBuf(){
        return this.requestBuffer;
    }
​
    public int write(ByteBuffer byteBuffer) throws IOException{
        return socketChannel.write(byteBuffer);
    }
​
    public void close() throws IOException{
        socketChannel.close();
    }
}

ChannelIO主要的几个作用就是读和写,默认的ByteBuffer为4096,但是报文大小有可能超过,所以这里我们读之前看看需不需要自动扩容,这个类被请求处理者所处理, 请求处理者要负责解析HTTP报文,HTTP报文有请求方式,有结束标志,有uri,这个解析的任务我们放在Request这个类来处理:

vbnet 复制代码
public class Request {
​
    private Action action;
​
    private URI uri;
​
    private String version;
​
    public Request(Action action, URI uri, String version) {
        this.action = action;
        this.uri = uri;
        this.version = version;
    }
​
    static class Action{
        private String name;
​
        static Action GET = new Action("GET");
​
        static Action POST = new Action("POST");
​
        static Action PUT = new Action("PUT");
​
        static Action HEAD = new Action("HEAD");
​
        public Action(String name) {
            this.name = name;
        }
        public String toString(){
            return this.name;
        }
        static Action parse(String s){
            if ("GET".equals(s)){
                return GET;
            }
            if ("POST".equals(s)){
                return POST;
            }
            if ("PUT".equals(s)){
                return PUT;
            }
            if ("HEAD".equals(s)){
                return HEAD;
            }
            // 参数不合法
            throw new IllegalArgumentException(s);
        }
    }
​
​
    public  static boolean isComplete(ByteBuffer byteBuffer){
        int position = byteBuffer.position() - 4;
        if (position < 0){
            return false;
        }
        return byteBuffer.get(position + 0) == '\r'
                && byteBuffer.get(position + 1) == '\n'
                && byteBuffer.get(position + 2) == '\r'
                && byteBuffer.get(position + 3) == '\n';
    }
​
    private static Charset ascii = StandardCharsets.US_ASCII;
​
    /**
     * 正则表达式 用来分割请求报文
     * http 请求的报文是: GET /dir/file HTTP/1.1
     *  Host: hostname
     *  被正则表达式分割以后:
     *      group[1] = "GET"
     *      group[2] = "/dir/file"
     *      group[3] = "1.1"
     *      group[4] = "hostname"
     */
    private static Pattern requestPattern
            = Pattern.compile("\A([A-Z]+) +([^ ]+) +HTTP/([0-9\.]+)$"
                    + ".*^Host: ([^ ]+)$.*\r\n\r\n\z",
            Pattern.MULTILINE | Pattern.DOTALL);
​
    public static  Request parse(ByteBuffer byteBuffer) throws RequestException {
        // byte to char
        CharBuffer charBuffer = ascii.decode(byteBuffer);
        Matcher matcher = requestPattern.matcher(charBuffer);
        // 未匹配
        if (!matcher.matches()){
            throw new  RequestException();
        }
        Action a;
        try {
            a = Action.parse(matcher.group(1));
        }catch (IllegalArgumentException  e){
            throw new RequestException();
        }
        URI u = null;
        try {
            u = new URI("http://" + matcher.group(4) + matcher.group(2));
        }catch (URISyntaxException e){
           throw new RequestException(e);
        }
        return new Request(a,u,matcher.group(3));
    }
}

这个类主要封装HTTP报文的请求方式、版本、URI。有Request就有Reply,一个HTTP响应通常情况下会有状态码和内容,这里我们的HTTP服务器将来要扩展到各种类型:

java 复制代码
public interface Sendable {
    // 做转码
    void prepare() throws IOException;
    // 发送动作
    boolean send(ChannelIO channelIO);
    
    void release();
}
public interface Content extends Sendable {
    // 发送类型
    String type();
    
    // 长度
    long length();
}
public class StringContent implements Content{
​
    private String type;    // MIME type
​
    private String content;
​
    private ByteBuffer byteBuffer;
​
    private static final Charset ascii = StandardCharsets.US_ASCII;
​
    StringContent(CharSequence c, String t) {
        content = c.toString();
        type = t + "; charset=iso-8859-1";
    }
​
    StringContent(CharSequence c) {
        this(c, "text/plain");
    }
​
    StringContent(Exception x) {
        StringWriter sw = new StringWriter();
        x.printStackTrace(new PrintWriter(sw));
        type = "text/plain; charset=iso-8859-1";
        content = sw.toString();
    }
​
    @Override
    public String type() {
        return type;
    }
​
    @Override
    public long length() {
        return byteBuffer.remaining();
    }
​
    @Override
    public void prepare() throws IOException {
        encode();
        // 在写入之前就需要调用一下rewind方法
        byteBuffer.rewind();
    }
​
    private void encode() {
        if (byteBuffer == null){
            byteBuffer =  ascii.encode(CharBuffer.wrap(content));
        }
    }
​
    @Override
    public boolean send(ChannelIO channelIO) throws IOException {
        if (byteBuffer == null)
            throw new IllegalStateException();
        // 写的时候 不见得一次写完
        channelIO.write(byteBuffer);
        // hasRemaining 代表是否还有剩余
        // 如果有剩余就可以写着写
        return  byteBuffer.hasRemaining();
    }
​
    /**
     * 这是个空方法
     * 后面只是为了统一调用
     */
    @Override
    public void release() {
​
    }
}
java 复制代码
public class Reply implements Sendable {
​
    static class Code{
​
        private int number;
​
        private String description;
​
        public Code(int number, String description) {
            this.number = number;
            this.description = description;
        }
​
        static Code OK = new Code(200,"OK");
​
        static Code BAD_REQUEST = new Code(400,"Bad Request");
​
        static Code NOT_FOUND = new Code(404,"Not Found");
​
        static Code METHOD_NOT_ALLOWED = new Code(405,"Method Not Allowed");
    }
​
    private Code code;
​
    private Content content;
​
    private ByteBuffer headerBuffer;
​
​
    private static String CRLF = "\r\n";
​
    private static Charset ascii = Charset.forName("US-ASCII");
​
    public Reply(Code code, Content content) {
        this.code = code;
        this.content = content;
    }
​
    /**
     * 这个方法负责添加请求头
     * @return
     */
    private ByteBuffer headers(){
        CharBuffer cb = CharBuffer.allocate(1024);
        cb.put("HTTP/1.0 ").put(code.toString()).put(CRLF);
        cb.put("Server: niossl/0.1").put(CRLF);
        cb.put("Content-type: ").put(content.type()).put(CRLF);
        cb.put("Content-length: ")
                .put(Long.toString(content.length())).put(CRLF);
        cb.put(CRLF);
        cb.flip();
        return ascii.encode(cb);
    }
​
    @Override
    public void prepare() throws IOException {
        content.prepare();
        headerBuffer = headers();
    }
​
    @Override
    public boolean send(ChannelIO channelIO) throws IOException {
         // 先写请求头
        if (headerBuffer.hasRemaining()){
            if (channelIO.write(headerBuffer) <= 0)
                return true;
        }
        // 再写响应内容
        if (content.send(channelIO))
            return true;
        return false;
    }
​
    @Override
    public void release() {
        content.release();
    }
}

连接建立之后开始提取数据:

ini 复制代码
public class RequestServicer implements Runnable {
​
​
    private ChannelIO channelIO;
​
    public RequestServicer(ChannelIO channelIO) {
        this.channelIO = channelIO;
    }
​
    private void service() throws IOException {
        ByteBuffer byteBuffer = receive(); // 接收数据
        Request request = null;
        Reply reply = null;
        try {
            request = Request.parse(byteBuffer);
        } catch (RequestException e) {
            reply = new Reply(Reply.Code.BAD_REQUEST, new StringContent(e));
        }
        // 说明正常解析
        if (reply == null) {
            reply  = build(request); // 构建回复
        }
        reply.prepare();
        do {} while (reply.send(channelIO));         // Send
    }
​
    @Override
    public void run() {
        try {
            service();
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
​
    
    ByteBuffer receive() throws IOException {
        for (; ; ) {
            int read = channelIO.read();
            ByteBuffer bb = channelIO.getReadBuf();
            if ((read < 0) || (Request.isComplete(bb))) {
                bb.flip();
                return bb;
            }
        }
    }
​
    Reply build(Request rq) throws IOException {
​
        Reply rp = null;
        Request.Action action = rq.action();
        if ((action != Request.Action.GET)) {
            rp = new Reply(Reply.Code.METHOD_NOT_ALLOWED,
                    new StringContent(rq.toString()));
            rp.prepare();
            return rp;
        }
        rp = new Reply(Reply.Code.OK,
                new StringContent("hello world"));
        rp.prepare();
        return rp;
    }
}

NIO简介

上面模型也被称为BIO模型,也就是Blocking Input/Output, 其实上面已经分析出来在哪里了,也就是读数据的时候未必可以读,但是我们的read调用就被阻塞在那里,我们自然能够想到能否让操作系统为我们提供一个非阻塞的read函数, 这个 read 函数的效果是,如果没有数据到达时(到达网卡并拷贝到了内核缓冲区),立刻返回一个错误值(-1),而不是阻塞地等待。操作系统提供了这样的功能,只需要在调用 read 前,将文件描述符设置为非阻塞即可。这样我们在线程里面调用read函数,直到返回值不为-1的,再开始处理业务。 但是在数据到达内核缓冲区,这个阶段仍然是阻塞的,需要等待数据从内核缓冲区拷贝到用户缓冲区,才能返回。

这里可以为每个连接准备一个线程来处理,这其实也是解决问题的方案,一些连接请求不多的HTTP服务器现在还是这么处理的,那么对于连接过多的,多线程就有些乏力了,当然也可以有聪明的方法,我们可以每accept一个连接之后,将这个文件描述符(可以理解为Socket的引用)放在一个数组里面,然后弄一个新的线程去不断遍历这个数组,调用每一个元素的非阻塞 read 方法,这样,我们就成功用一个线程处理了多个客户端连接。

这看起来就有多路复用的意思了,但这和我们用多线程去将阻塞 IO 改造成看起来是非阻塞 IO 一样,这种遍历方式也只是我们用户自己想出的小把戏,每次遍历遇到 read 返回 -1 时仍然是一次浪费资源的系统调用。所以,还是得恳请操作系统老大,提供给我们一个有这样效果的函数,我们将一批文件描述符通过一次系统调用传给内核,由内核层去遍历,才能真正解决这个问题。

select 是操作系统提供的系统调用函数,通过它,我们可以把一个文件描述符的数组发给操作系统, 让操作系统去遍历,确定哪个文件描述符可以读写, 然后告诉我们去处理。

但是这个函数仍然不完美,原因在于:

  1. select 调用需要传入 fd 数组,需要拷贝一份到内核,高并发场景下这样的拷贝消耗的资源是惊人的。(可优化为不复制)
  2. select 在内核层仍然是通过遍历的方式检查文件描述符的就绪状态,是个同步过程,只不过无系统调用切换上下文的开销。(内核层可优化为异步事件通知)
  3. select 仅仅返回可读文件描述符的个数,具体哪个可读还是要用户自己遍历。(可优化为只返回给用户就绪的文件描述符,无需用户做无效的遍历)

但也不是不能用,但select还有限制,这个限制就是select 只能监听 1024 个文件描述符的限制,后面的poll去掉了这个限制。最终解决select函数的大boss叫epoll,针对select函数的三个不完美的点进行了修复:

  1. 内核中保存一份文件描述符集合,无需用户每次都重新传入,只需告诉内核修改(添加、修改、监控的文件描述符)的部分即可。
  2. 内核不再通过轮询的方式找到就绪的文件描述符,而是通过异步 IO 事件唤醒。
  3. 内核仅会将有 IO 事件的文件描述符返回给用户,用户也无需遍历整个文件描述符集合。

重回NIO

"内核仅会将有IO事件的文件描述符返回给用户",仔细读这一句话,我以为是select函数的返回值是一个集合,但是我去看了一下这个函数:

arduino 复制代码
int select(int maxfd, fd_set *readfds, fd_set *writefds,fd_set *exceptfds, struct timeval *timeout);

fd是一个集合类型,从参数名字上我们来推断readfds传入需要监视读事件的文件描述符,writefds是需要监视读事件的文件描述符,exceptfds是异常事件的文件描述符,这里我们提到了文件描述符,这个文件描述符是用来代表一个打开的文件、或者socket、或者其他数据源。定义了能对该文件做的操作。当select函数有所返回的时候,会修改传入的集合。select函数是系统调用,Java层面对应的抽象也就是Selector,使用起来倒是简单:

ini 复制代码
Selector selector = Selector.open();
selector.select();

默认选择当前操作系统的实现,我们看下Open的实现,我的电脑装的操作系统是Windows,select的实现在Oracle 的hotspot VM中是闭源的,观察他的实现要在OpenJDK上,我这里随手选了一个JDK 11版本的实现:

那怎么让这个选择器知道我对某个事件感兴趣 , 读事件就绪、写事件就绪其实通道(通道也就是对连接的抽象)上发生的事件,按照我之前的想法Selector这个类里面应该会有一个register之类的方法,但是没找到,不在Selector就在ServerSocketChannel里面,果然我在ServerSocketChannel找到了register方法, 这个register来自AbstractSelectableChannel。

java 复制代码
// ops是一个枚举值,att是对应的事件触发之后,交付给哪个对象处理
public final SelectionKey register(Selector sel, int ops, Object att)

所以我们可以写成下面这样:

ini 复制代码
ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
serverSocketChannel.bind(new InetSocketAddress(8080),1024);
serverSocketChannel.configureBlocking(false);
Selector selector = Selector.open();
Runnable runnable = ()-> {
    System.out.println("hello world");
};
serverSocketChannel.register(selector,SelectionKey.OP_ACCEPT,runnable);
while (true){
        selector.select();
        Set<SelectionKey> selectKeys = selector.selectedKeys();
        Iterator<SelectionKey> iterator = selectKeys.iterator();
        while(iterator.hasNext()) {
            SelectionKey selectionKey = iterator.next();
            Runnable handle = (Runnable)selectionKey.attachment();
            handle.run();
            iterator.remove();
        }
}

解释一下这段代码,我们首先将ServerSocketChannel变成非阻塞模式 , 然后绑定监听的端口,打开一个选择器,然后通过register方法将ServerSocketChannel纳入Selector的管辖范围之内之内,告知选择器这个Channel对连接就绪这个事件感兴趣,然后调用Selector的select方法,这个方法上直到有就绪事件之前一直会阻塞,然后从选择器上获取SelectionKey集合,那么该如何理解这个SelectionKey呢? 我们翻下SelectionKey的注释:

A token representing the registration of a SelectableChannel with a Selector.

表示Selector和SelectableChannel注册的标记

这个SelectableChannel是何方神圣,我们翻阅一下:

SocketChannel和ServerSocketChannel都继承了这个类,上面我们使用的register就来自于这个类, 现在我们大致能理清楚Selector和这两个Channel之间的关系:

连接建立之后我们可以通过ServerSocketChannel的accept方法拿到SocketChannel,通过SocketChannel可以读写数据,但是在NIO下面读未必是就绪的,在SelectableChannel里面提供了向选择器注册感兴趣事件的方法,而ServerSocketChannel和SocketChannel又继承SelectableChannel。

A selection key is created each time a channel is registered with a selector.

每次通道注册到Selector的时候都会创建一个SelectionKey

SelectionKey关联了Channel,Selector向我们返回连接就绪,可读可写的时候,返回一个SelectionKey,然后我们通过这个SelectionKey去拿注册的对象和通道,然后接着向选择器注册可读、可写事件。

那么对于TCP协议来说,读和连接就绪相对不可控一些,需要客户端主动发起,那么写就绪呢,一个Socket创建之后,操作系统会为他分配资源,相对写比较可控,那什么时候写呢,原则上我们在连接建立之后就可以马上写,如果写到内核的缓冲区满了不能写,我们再向选择器注册写事件,注意写完资源要马上取消对这个事件感兴趣,要不然选择器上就一直会有就绪的写事件返回出来。我们回想一下HTTP是收完客户端发完的报文,再写,那么对于WebSocket来说连接建立,双方就能随时写。本篇我们写是一个HTTP服务器,所以我们设计的是在读完之后再注册对写事件感兴趣。

A key remains valid until it is cancelled by invoking its cancel method, by closing its channel, or by closing its selector.

这个Key会一直有效,直到调用cancel方法,关闭其通道,关闭选择器。

Cancelling a key does not immediately remove it from its selector; it is instead added to the selector's cancelled-key set for removal during the next selection operation.

当你选择从Selector中取消一个键的时候,这个键不会立刻从它所属的Selector中被移除,相反,这个取消的键会被加入到已取消的集合中,下一次选择操作的时候,才会被移除。

The validity of a key may be tested by invoking its isValid method.

可以通过调用键的 isValid 方法来测试键的有效性。

上面我们给的示例,事实上会一直输出Hello World,原因在于我们没有处理这个连接,也就是调用对应的accept方法,所以Selector上一直都会有就绪的事件。理解了这一步,我们就能给出NIO的实现了,首先向Selector注册对连接就绪事件感兴趣,连接就绪之后,我们向Selector注册对读事件感兴趣,在接收玩HTTP报文之后,我们就可以写,如果写不完,我们可以向选择器上注册对写事件就绪,就绪的时候调用对应的处理者:

java 复制代码
// 这里负责注册
public interface Dispatcher extends Runnable {
    void register(SelectableChannel selectableChannel, int ops, HttpHandler httpHandler) throws ClosedChannelException;
}
​
public class Dispatcher1 implements Dispatcher{
    private Selector selector;
    
    public Dispatcher1() throws IOException {
        selector = Selector.open();
    }
​
    @Override
    public void register(SelectableChannel selectableChannel, int ops, HttpHandler httpHandler) throws ClosedChannelException {
        selectableChannel.register(selector,ops,httpHandler);
    }
​
    @Override
    public void run() {
        // 无限循环分发事件
        for(;;){
            try {
                dispatch();
            } catch (IOException e) {
                e.printStackTrace();
            }
        }
    }
​
    private void dispatch() throws IOException {
       // 阻塞到至少有一个就绪事件
        selector.select();
        for (Iterator<SelectionKey> iterator = selector.selectedKeys().iterator(); iterator.hasNext();){
            SelectionKey selectionKey = iterator.next();
            iterator.remove();
            // 获取对应的register方法
            HttpHandler httpHandler = (HttpHandler)selectionKey.attachment();
            // 交给对应的处理者处理
            httpHandler.handler(selectionKey);
        }
    }
}
// 这里用接口是为了统一处理
public interface HttpHandler {
    void handler(SelectionKey selectionKey);
}
public class AcceptHandler implements HttpHandler{
​
    private ServerSocketChannel serverSocketChannel;
​
    private Dispatcher dispatcher;
​
    public AcceptHandler(ServerSocketChannel serverSocketChannel, Dispatcher dispatcher) {
        this.serverSocketChannel = serverSocketChannel;
        this.dispatcher = dispatcher;
    }
​
    @Override
    public void handler(SelectionKey selectionKey) throws IOException {    
        SocketChannel socketChannel = serverSocketChannel.accept();     
        // 注册对读事件感兴趣
            dispatcher.register(socketChannel,SelectionKey.OP_READ,new questHandler(dispatcher,ChannelIO.getInstance(socketChannel,false)));
​
    }
}
public class NonBlockingServer extends Server{
    public NonBlockingServer(int port, int backlog, boolean secure) throws IOException {
        super(port, backlog, secure);
        serverSocketChannel.configureBlocking(false);
    }
​
    @Override
    protected void runServer() throws IOException {
        Dispatcher dispatcher = new Dispatcher1();
        AcceptHandler acceptHandler = new AcceptHandler(serverSocketChannel,dispatcher);
        dispatcher.register(serverSocketChannel, SelectionKey.OP_ACCEPT,acceptHandler);
        dispatcher.run();
    }
}

接下来让我们来想想这个RequestHandler的逻辑应该是什么,我们做的是一个HTTP Server,所以理应是先接收报文,等报文接收完毕,再解析报文,报文解析之后构建响应,写回复。

java 复制代码
public class RequestHandler implements HttpHandler{
​
    private Dispatcher dispatcher;
​
    private ChannelIO channelIO;
​
    private Request request = null;
​
    private boolean requestReceived;
​
    private ByteBuffer rbb = null;
​
    private Reply reply;
​
    public RequestHandler(Dispatcher dispatcher, ChannelIO channelIO) {
        this.dispatcher = dispatcher;
        this.channelIO = channelIO;
    }
​
    /**
     * 先接收数据,后解析, 然后是发送
     * @param selectionKey
     */
    @Override
    public void handler(SelectionKey selectionKey) throws IOException {
        SelectableChannel  selectableChannel  = selectionKey.channel();
        if (request == null){
            if (!received()){
                return;
            }
            rbb = channelIO.getReadBuf();
            rbb.flip();
​
            if (requestParse(rbb)){
                reply = replyBuild();
            }
            reply.prepare();
            // 开始构建回复
            if (replySend()){
                dispatcher.register(selectableChannel,SelectionKey.OP_WRITE,new ResponseHandler());
            }
            channelIO.close();
            if (reply != null){
                reply.release();
            }
​
        }
    }
    private boolean replySend() {
        try {
            return  reply.send(channelIO);
        } catch (IOException e) {
            System.out.println("处理异常");
            return false;
        }
    }
​
    private Reply replyBuild() throws IOException {
        Reply rp = null;
        Request.Action action = request.action();
        if ((action != Request.Action.GET)) {
            rp = new Reply(Reply.Code.METHOD_NOT_ALLOWED,
                    new StringContent(request.toString()));
            rp.prepare();
            return rp;
        }
        rp = new Reply(Reply.Code.OK,
                new StringContent("hello world"));
        rp.prepare();
        return rp;
​
    }
​
    private boolean requestParse(ByteBuffer rbb) {
        try {
            request =  Request.parse(rbb);
            return true;
        } catch (RequestException e) {
            reply = new Reply(Reply.Code.BAD_REQUEST,
                    new StringContent("解析异常"));
        }
        return false;
    }
​
    private boolean received() throws IOException {
​
        if (requestReceived) {
            return true;
        }
        // 读取完毕,将对应
        if ((channelIO.read() < 0) || Request.isComplete(channelIO.getReadBuf())){
            return requestReceived = true;
        }
        return false;
    }
}
public class ResponseHandler implements HttpHandler{
​
    /**
     *  只是完善这个模型,我们的写一次就到位了
     * @param selectionKey
     * @throws IOException
     */
    @Override
    public void handler(SelectionKey selectionKey) throws IOException {
        System.out.println();
        selectionKey.cancel();
    }
}

到现在为止我们用Java 的Socket API 已经基本构建了一个NIO模型的HTTP服务器,这里没有一股脑将相关API全部列在这里,我选择的风格是我喜欢教科书的风格,用到什么讲什么,一个应用服务器整体是不断处理请求的,所以我们做了一个Dispatch,用来分发Selector上的就绪事件,先是注册对连接就绪这个事件感兴趣的处理器我们调用的是SelectableChannel中的register方法:

c 复制代码
register(Selector sel, int ops, Object att)

第三个参数在事件被触发的时候,我们可以用SelectionKey的attachment拿到这个对象,为了统一调用我们让对应都实现了HttpHandler接口。注意到现在为止我们只有一个线程在处理所有的事件,这可以做的很快,如果是纯内存操作的话,但是如果写入的内容有些大导致,其他连接就可能就要排队,这无疑是我们不想看到的,所以上面的模型我们可以对其进行改造,将接受连接的处理器,单独启用一个线程来处理,后面读写的挂一个线程池来处理。这就是单线程的Reactor模式和多线程的Reactor模式的雏形。

NIO解析

所谓的BIO,Blocking在read上面,读数据的时候未必还可以读,数据的流转需要一个过程,但是我们不希望被阻塞在这里,我们自然能够想到,能否让操作系统为我们提供一个非阻塞的read函数,这个read函数的效果是,如果没有数据到达时,立刻返回一个值,而不是立刻等待。

当然我们可以为每一个连接准备一个线程来处理,这也不是不能用,现在来说一些应用服务器还有停留在BIO模式的,但对于连接过多的,我们就需要考虑别的解决方案了,当然也有聪明的方法,我们可以每accept一个连接之后,将这个文件描述符放在一个数组里面,然后弄一个新的线程去不断遍历这个数组,调用每一个元素的非阻塞read方法,这样我们就可以用一个线程来处理多个客户端的连接。

这看起来就有点多路复用的意思了,但是这只是用户态的NIO,我们仍然需要发起系统调用,系统调用相当昂贵,所以我们得恳求操作系统老大,提供给我们一个这样的函数,我们将一批文件描述符通过一次调用传给内核,由内核层去遍历,才真正去解决这个问题。

select是Linux提供的系统调用函数,通过它,我们可以把一个文件描述符的数组发给操作系统,让操作系统去遍历,确定哪个文件描述可以读写,然后告诉我们去处理。但是这个函数仍然不完美,原因在于:

  1. select 调用需要传入 fd 数组,需要拷贝一份到内核,高并发场景下这样的拷贝消耗的资源是惊人的。(可优化为不复制)
  2. select 在内核层仍然是通过遍历的方式检查文件描述符的就绪状态,是个同步过程,只不过无系统调用切换上下文的开销。(内核层可优化为异步事件通知)
  3. select 仅仅返回可读文件描述符的个数,具体哪个可读还是要用户自己遍历。(可优化为只返回给用户就绪的文件描述符,无需用户做无效的遍历)

但也不是不能用,但select还有限制,这个限制就是select 只能监听 1024 个文件描述符的限制,后面的poll去掉了这个限制。最终解决select函数的大boss叫epoll,针对select函数的三个不完美的点进行了修复:

  1. 内核中保存一份文件描述符集合,无需用户每次都重新传入,只需告诉内核修改(添加、修改、监控的文件描述符)的部分即可。
  2. 内核不再通过轮询的方式找到就绪的文件描述符,而是通过异步 IO 事件唤醒。
  3. 内核仅会将有 IO 事件的文件描述符返回给用户,用户也无需遍历整个文件描述符集合。

总结一下

我们总结一下构建一个应用服务器的要点,也就是管理连接、读取数据、构建回复,这是三个最核心的问题,那么我们自然能够想到能不能将这些封装起来,我们只用关心处理的逻辑就行,这也就是我们将在下一篇讲的Netty,上面我们我们没有提到拆包的概念,原因我认为这是应用层报文需要考虑的问题,报文的结束标志位是什么。这篇文章写了一段时间,倒是有点超出意料,每次写感觉都对网络的理解更深刻了一些,然后看不清的东西又多了一些。

参考资料

[1] what happens after read is called for a Linux socket stackoverflow.com/questions/1...

[2] What is the difference between the kernel space and the user space? stackoverflow.com/questions/5...

[3] What is difference between User space and Kernel space? unix.stackexchange.com/questions/8...

[4] Linux网络数据包接受过程 simonzgx.github.io/2020/08/17/...

[5] Kernel Mode Definition www.linfo.org/kernel_mode...

[6] What are high memory and low memory on Linux? unix.stackexchange.com/questions/4...

[7] Implementing System Calls www.cs.swarthmore.edu/~kwebb/cs45...

[8] LinuxSystemCalls.pdf comet.lehman.cuny.edu/jung/cmp426...

[9] The Operating System www.cs.swarthmore.edu/~kwebb/cs31...

[10] Difference between System call and System call service routines stackoverflow.com/questions/7...

[11] what is "java.net.SocketException: Connection reset" learn.redhat.com/t5/General/...

[12] What does "connection reset by peer" mean? stackoverflow.com/questions/1...

[13] TCP: Differences Between FIN and RST www.baeldung.com/cs/tcp-fin-...

[14] FIN vs RST in TCP connections stackoverflow.com/questions/1...

[15] TCP学习笔记(二) 相识篇 juejin.cn/post/710309...

[16] TCP-4-times-close wiki.wireshark.org/TCP-4-times...

[18] What does "connection reset by peer" mean? stackoverflow.com/questions/1...

[19] linux select函数解析以及事例 zhuanlan.zhihu.com/p/57518857

相关推荐
码农小旋风23 分钟前
详解K8S--声明式API
后端
Peter_chq25 分钟前
【操作系统】基于环形队列的生产消费模型
linux·c语言·开发语言·c++·后端
Yaml41 小时前
Spring Boot 与 Vue 共筑二手书籍交易卓越平台
java·spring boot·后端·mysql·spring·vue·二手书籍
小小小妮子~1 小时前
Spring Boot详解:从入门到精通
java·spring boot·后端
hong1616881 小时前
Spring Boot中实现多数据源连接和切换的方案
java·spring boot·后端
睡觉谁叫~~~2 小时前
一文解秘Rust如何与Java互操作
java·开发语言·后端·rust
2401_865854884 小时前
iOS应用想要下载到手机上只能苹果签名吗?
后端·ios·iphone
AskHarries5 小时前
Spring Boot集成Access DB实现数据导入和解析
java·spring boot·后端
2401_857622665 小时前
SpringBoot健身房管理:敏捷与自动化
spring boot·后端·自动化
程序员阿龙5 小时前
基于SpringBoot的医疗陪护系统设计与实现(源码+定制+开发)
java·spring boot·后端·医疗陪护管理平台·患者护理服务平台·医疗信息管理系统·患者陪护服务平台