zoukankan      html  css  js  c++  java
  • java网络编程总结

      OSI网络七层模型

      为了不同计算机厂商生产的电脑能够通信,以便在更大的范围内建立计算机网络,有必要建立一个国际范围内的网络体系结构标准,也就有了OSI网络七层模型

      TCP是一个重要的传输层协议,提供面向连接、可靠、有序的字节流传输服务,在传输数据前必须先建立tcp连接,传输报文如下:

      tcp三次握手和四次挥手

      UDP协议提供无连接、不可靠的数据包尽力传输的服务

      BIO阻塞式网络编程

      bio网络编程基于socket和io,先看一个简单的client程序和server端程序

    package com.example.test.BIO;
    
    import java.io.*;
    import java.net.Socket;
    import java.util.Scanner;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class BioClient {
    
        public static void main(String[] args) throws IOException {
            Socket socket = new Socket("localhost",8080);
            OutputStream outputStream = socket.getOutputStream();
            BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(outputStream,"utf-8"));
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入:");
            String msg = scanner.nextLine();
            bufferedWriter.write(msg);
            scanner.close();
            socket.close();
        }
    }
    package com.example.test.BIO;
    
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.InputStreamReader;
    import java.net.ServerSocket;
    import java.net.Socket;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class BioServer {
    
        public static void main(String[] args) throws Exception {
    
            ServerSocket serverSocket = new ServerSocket(8080);
            System.out.println("server启动");
            while(!serverSocket.isClosed()){
                Socket socket = serverSocket.accept();
                try {
                    InputStream in = socket.getInputStream();
                    BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(in));
                    String msg = null;
                    while ((msg = bufferedReader.readLine()) != null) {
                        if (msg.length() == 0) {
                            break;
                        }
                        System.out.println(msg);
                    }
                    System.out.println("收到消息,来自:" + socket.toString());
                } catch (IOException e) {
                    e.printStackTrace();
                } finally {
                    socket.close();
                }
            }
            serverSocket.close();
    
        }
    }

      由于serverSocket.accept()及bufferedReader.readLine()都是阻塞的方法,因此server端是一个单线程的,不能满足多个client同时请求,改造server端,使其支持多线程

    package com.example.test.BIO;
    
    import java.io.BufferedReader;
    import java.io.IOException;
    import java.io.InputStream;
    import java.io.InputStreamReader;
    import java.net.ServerSocket;
    import java.net.Socket;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class BioServer2 {
    
        private static ExecutorService threadPool = Executors.newCachedThreadPool();
    
        public static void main(String[] args) throws Exception {
    
            ServerSocket serverSocket = new ServerSocket(8080);
            System.out.println("server启动");
            while(!serverSocket.isClosed()){
                Socket socket = serverSocket.accept();
                System.out.println("收到新的请求" + socket.toString());
                threadPool.execute(()->{
                    try {
                        InputStream in = socket.getInputStream();
                        String msg = null;
                        while ((msg = bufferedReader.readLine()) != null) {
                            if (msg.length() == 0) {
                                break;
                            }
                            System.out.println(msg);
                        }
                        System.out.println("收到消息,来自:" + socket.toString());
                    } catch (IOException e) {
                        e.printStackTrace();
                    } finally {
                        try {
                            socket.close();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
                });
    
            }
            serverSocket.close();
    
        }
    }

      此时sever端的并发数就受限于线程池的大小,那么浏览器和我们server端如何交互呢,此时就需要对http协议有基础的认识

      http请求数据包解析

      http响应数据包解析

      http响应码状态

      对现有的server端程序改造,使其支持http请求

    package com.example.test.BIO;
    
    import java.io.*;
    import java.net.ServerSocket;
    import java.net.Socket;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class BioServer3 {
    
        private static ExecutorService threadPool = Executors.newCachedThreadPool();
    
        public static void main(String[] args) throws Exception {
    
            ServerSocket serverSocket = new ServerSocket(8080);
            System.out.println("server启动");
            while(!serverSocket.isClosed()){
                Socket socket = serverSocket.accept();
                System.out.println("收到新的请求" + socket.toString());
                threadPool.execute(()->{
                    try {
                        InputStream in = socket.getInputStream();
                        BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(in,"utf-8"));
                        String msg = null;
                        while ((msg = bufferedReader.readLine()) != null) {
                            if (msg.length() == 0) {
                                break;
                            }
                            System.out.println(msg);
                        }
                        System.out.println("收到消息,来自:" + socket.toString());
    
    
                        OutputStream outputStream = socket.getOutputStream();
                        outputStream.write("HTTP/1.1 200 OK
    ".getBytes());
                        outputStream.write("Content-Length: 11
    
    ".getBytes());
                        outputStream.write("Hello World".getBytes());
                        outputStream.flush();
                        bufferedReader.close();
                        outputStream.close();
                    } catch (IOException e) {
                        e.printStackTrace();
                    } finally {
                        try {
    
                            socket.close();
                        } catch (IOException e) {
                            e.printStackTrace();
                        }
                    }
                });
    
            }
            serverSocket.close();
    
        }
    }

      改造client端,使其能够接受server端的响应

    package com.example.test.BIO;
    
    import java.io.*;
    import java.net.Socket;
    import java.util.Scanner;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class BioClient {
    
        public static void main(String[] args) throws IOException {
            Socket socket = new Socket("localhost",8080);
            OutputStream outputStream = socket.getOutputStream();
            BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(outputStream,"utf-8"));
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入:");
            String msg = scanner.nextLine();
            bufferedWriter.write(msg);
            bufferedWriter.flush();
            socket.shutdownOutput();
            InputStream inputStream =socket.getInputStream();
            BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(inputStream));
            String returnMsg = null;
            while((returnMsg = bufferedReader.readLine())!=null){
                if(returnMsg.length()==0){
                    break;
                }
                System.out.println(returnMsg);
            }
            scanner.close();
            socket.close();
        }
    }

      阻塞式编程相关概念

      NIO网络编程

      从java1.4开始,出现了新的JAVA IO非阻塞API,NIO有三个核心概念:Buffer缓冲区、Channel通道和Selector选择器

      Buffer缓冲区

      demo如下

    package com.example.test.NIO;
    
    import java.nio.ByteBuffer;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionsdf
     */
    public class BufferDemo {
    
        public static void main(String[] args) {
            ByteBuffer byteBuffer = ByteBuffer.allocate(4);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            byteBuffer.put((byte) 1);
            byteBuffer.put((byte) 2);
            byteBuffer.put((byte) 3);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            System.out.println("开始读取");
            //切换为读模式
            byteBuffer.flip();
            byte a = byteBuffer.get();
            System.out.println(a);
            byte c = byteBuffer.get();
            System.out.println(c);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            //清除已读取的数据,转为写模式
            byteBuffer.compact();
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
    
            byteBuffer.put((byte) 3);
            byteBuffer.put((byte) 4);
            byteBuffer.put((byte) 5);
            System.out.println(String.format("最终的情况,capacity容量:%s, position位置:%s, limit限制:%s", byteBuffer.capacity(),
                    byteBuffer.position(), byteBuffer.limit()));
    
        }
    }
    package com.example.test.NIO;
    
    import java.nio.ByteBuffer;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class DerictBufferDemo {
    
        public static void main(String[] args) {
            ByteBuffer byteBuffer = ByteBuffer.allocateDirect(4);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            byteBuffer.put((byte) 1);
            byteBuffer.put((byte) 2);
            byteBuffer.put((byte) 3);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            System.out.println("开始读取");
            //切换为读模式
            byteBuffer.flip();
            byte a = byteBuffer.get();
            System.out.println(a);
            byte c = byteBuffer.get();
            System.out.println(c);
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
            //清除已读取的数据,转为写模式
            byteBuffer.compact();
            System.out.println(String.format("初始化容量:%s,position位置:%s,limit限制:%s",
                    byteBuffer.capacity(),byteBuffer.position(),byteBuffer.limit()));
    
            byteBuffer.put((byte) 3);
            byteBuffer.put((byte) 4);
            byteBuffer.put((byte) 5);
            System.out.println(String.format("最终的情况,capacity容量:%s, position位置:%s, limit限制:%s", byteBuffer.capacity(),
                    byteBuffer.position(), byteBuffer.limit()));
    
        }
    }

      Channel通道

      利用nio构建简单的client程序和server端程序

    package com.example.test.NIO;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.SocketChannel;
    import java.util.Scanner;
    
    public class NioClient {
    
        public static void main(String[] args) throws IOException {
    
            SocketChannel socketChannel =SocketChannel.open();
            socketChannel.configureBlocking(false);
            socketChannel.connect(new InetSocketAddress("127.0.0.1",8080));
            while(!socketChannel.finishConnect()){
                //没连接上则一直连接
                Thread.yield();
            }
            Scanner scanner = new Scanner(System.in);
            System.out.println("请输入:");
            String msg = scanner.nextLine();
            ByteBuffer byteBuffer = ByteBuffer.wrap(msg.getBytes());
            while(byteBuffer.hasRemaining()){
                socketChannel.write(byteBuffer);
            }
    
            System.out.println("收到服务端的响应:");
            ByteBuffer returnBuffer = ByteBuffer.allocate(1024);
            while(socketChannel.isOpen()&& socketChannel.read(returnBuffer)!=-1){
                if(returnBuffer.position()>0){
                    break;
                }
            }
            returnBuffer.flip();
            System.out.println(returnBuffer.limit());
            byte[] content = new byte[returnBuffer.limit()];
            returnBuffer.get(content);
            System.out.println(new String(content));
    
            scanner.close();
            socketChannel.close();
        }
    }
    package com.example.test.NIO;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.ServerSocketChannel;
    import java.nio.channels.SocketChannel;
    
    public class NioServer {
    
        public static void main(String[] args) throws IOException {
            //创建网络服务
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
            serverSocketChannel.configureBlocking(false);//设置为非阻塞
            serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口
            System.out.println("启动成功");
            while(true){
                SocketChannel socketChannel = serverSocketChannel.accept();//获取tcp连接通道
                if(socketChannel!=null){
                    System.out.println("收到新连接" + socketChannel.getRemoteAddress());
                    socketChannel.configureBlocking(false);//设置为非阻塞
                    ByteBuffer requestByte = ByteBuffer.allocate(1024);
                    while(socketChannel.isOpen()&& socketChannel.read(requestByte)!=-1){
                        //长连接情况下,需要手动判断数据是否读取完毕,此处做一个简单判断,position>0表示读完
                        if(requestByte.position()>0){
                            break;
                        }
                    }
                    if(requestByte.position()==0){
                        continue;//没数据则不继续后续处理
                    }
                    requestByte.flip();
                    byte[] content = new byte[requestByte.limit()];
                    requestByte.get(content);
                    System.out.println(new String(content));
                    System.out.println("收到数据,来自" + socketChannel.getRemoteAddress());
                    String response = "HTTP/1.1 200 OK 
    " + "Content_Length: 11 
    
    " + "Hello Word";
                    ByteBuffer returnByteBuffer = ByteBuffer.wrap(response.getBytes());
                    while(returnByteBuffer.hasRemaining()){
                        socketChannel.write(returnByteBuffer);
                    }
                }
    
            }
        }
    }

      此时发现由于循环读取ByteBuffer中的数据,server只能同时接受一个请求,因此需要对server进行改造

    package com.example.test.NIO;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.ServerSocketChannel;
    import java.nio.channels.SocketChannel;
    import java.util.ArrayList;
    import java.util.Iterator;
    
    /**
     * @author hehang on 2019-05-24
     * @descriptionasd
     */
    public class NioServer1 {
    
        private static ArrayList<SocketChannel>  socketChannels = new ArrayList<>();
    
        public static void main(String[] args) throws IOException {
            //创建网络服务
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
            serverSocketChannel.configureBlocking(false);//设置为非阻塞
            serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口
            System.out.println("启动成功");
            while(true){
                SocketChannel socketChannel = serverSocketChannel.accept();//获取tcp连接通道
                if(socketChannel!=null) {
                    System.out.println("收到新连接" + socketChannel.getRemoteAddress());
                    socketChannel.configureBlocking(false);//设置为非阻塞
                    socketChannels.add(socketChannel);
                }else{
                    //没有新连接的时候,就去处理现有连接的数据
                    Iterator<SocketChannel> iterator = socketChannels.iterator();
                    while(iterator.hasNext()){
                        SocketChannel socketChannel1 = iterator.next();
                        ByteBuffer requestByte = ByteBuffer.allocate(1024);
                        if(socketChannel1.read(requestByte)==0){
                            continue;
                        }
                        while(socketChannel1.isOpen()&& socketChannel1.read(requestByte)!=-1){
                            //长连接情况下,需要手动判断数据是否读取完毕,此处做一个简单判断,position>0表示读完
                            if(requestByte.position()>0){
                                break;
                            }
                        }
                        requestByte.flip();
                        byte[] content = new byte[requestByte.limit()];
                        requestByte.get(content);
                        System.out.println(new String(content));
                        System.out.println("收到数据,来自" + socketChannel1.getRemoteAddress());
                        String response = "HTTP/1.1 200 OK 
    " + "Content_Length: 11 
    
    " + "Hello Word";
                        ByteBuffer returnByteBuffer = ByteBuffer.wrap(response.getBytes());
                        while(returnByteBuffer.hasRemaining()){
                            socketChannel1.write(returnByteBuffer);
                        }
                        iterator.remove();
                    }
    
                }
    
            }
        }
    }

      利用ByteBuffer和Channel已经可以进行NIO编程,但是这样却很不方便,JDK为我们提供一个新的API:Selector

      改造server端程序如下

    package com.example.test.NIO;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.SelectionKey;
    import java.nio.channels.Selector;
    import java.nio.channels.ServerSocketChannel;
    import java.nio.channels.SocketChannel;
    import java.util.Iterator;
    import java.util.Set;
    
    public class NioServer2 {
    
        public static void main(String[] args) throws IOException {
            //创建网络服务
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
            serverSocketChannel.configureBlocking(false);//设置为非阻塞
    
            //构建选择器,将serverSocketChannel注册上去,并且selector对serverSocketChannel上面的accept事件感兴趣
            Selector selector = Selector.open();
            SelectionKey selectionKey = serverSocketChannel.register(selector,0,serverSocketChannel);
            selectionKey.interestOps(SelectionKey.OP_ACCEPT);
    
            serverSocketChannel.socket().bind(new InetSocketAddress(8080));//绑定端口
            System.out.println("启动成功");
    
            while(true){
    
                selector.select();//该方法会阻塞,直到有事件通知才会返回
                //获取事件
                Set<SelectionKey> selectionKeys = selector.selectedKeys();
                Iterator<SelectionKey> iter = selectionKeys.iterator();
                while(iter.hasNext()){
    //                System.out.println("-------------");
                    SelectionKey key = iter.next();
                    iter.remove();
    
    
                    if(key.isAcceptable()){
                        ServerSocketChannel server = (ServerSocketChannel) key.attachment();
                        // 将拿到的客户端连接通道,注册到selector上面
                        SocketChannel clientSocketChannel = server.accept(); // mainReactor 轮询accept
                        clientSocketChannel.configureBlocking(false);
                        clientSocketChannel.register(selector, SelectionKey.OP_READ, clientSocketChannel);
                        System.out.println("收到新连接 : " + clientSocketChannel.getRemoteAddress());
                    }
    
                    if(key.isReadable()){
                        SocketChannel socketChannel = (SocketChannel) key.attachment();
                        try {
                            ByteBuffer requestBuffer = ByteBuffer.allocate(1024);
                            while (socketChannel.isOpen() && socketChannel.read(requestBuffer) != -1) {
                                // 长连接情况下,需要手动判断数据有没有读取结束 (此处做一个简单的判断: 超过0字节就认为请求结束了)
                                if (requestBuffer.position() > 0) break;
                            }
                            if(requestBuffer.position() == 0) continue; // 如果没数据了, 则不继续后面的处理
                            requestBuffer.flip();
                            byte[] content = new byte[requestBuffer.limit()];
                            requestBuffer.get(content);
                            System.out.println(new String(content));
                            System.out.println("收到数据,来自:" + socketChannel.getRemoteAddress());
                            // TODO 业务操作 数据库 接口调用等等
    
                            // 响应结果 200
                            String response = "HTTP/1.1 200 OK
    " +
                                    "Content-Length: 11
    
    " +
                                    "Hello World";
                            ByteBuffer buffer = ByteBuffer.wrap(response.getBytes());
                            while (buffer.hasRemaining()) {
                                socketChannel.write(buffer);
                            }
                        } catch (IOException e) {
                            // e.printStackTrace();
                            key.cancel(); // 取消事件订阅
                        }
                    }
                }
                selector.selectNow();
            }
    
        }
    }

      在上面的server端程序中,一个selector监听所有事件,一个线程处理所有请求事件,难以利用现代服务器多核特性,会成为性能瓶颈!,因此实际开发中要有多线程的运用,对此,JDK作者给出了NIO与多线程的结合使用:《Scalable IO in java》

    下面是多Reactor服务端实现

    package com.example.test.NIO;
    
    import java.io.IOException;
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.*;
    import java.util.Iterator;
    import java.util.Random;
    import java.util.Set;
    import java.util.concurrent.ExecutorService;
    import java.util.concurrent.Executors;
    import java.util.concurrent.FutureTask;
    import java.util.concurrent.LinkedBlockingQueue;
    import java.util.concurrent.atomic.AtomicInteger;
    
    /**
     * @author hehang on 2020-02-02
     * @description
     */
    public class NioServer3 {
    
        /** 处理业务操作的线程 */
        private static ExecutorService workPool = Executors.newCachedThreadPool();
    
        /**
         * 封装了selector.select()等事件轮询的代码
         */
        abstract class ReactorThread extends Thread {
    
            Selector selector;
            LinkedBlockingQueue<Runnable> taskQueue = new LinkedBlockingQueue<>();
    
            /**
             * Selector监听到有事件后,调用这个方法
             */
            public abstract void handler(SelectableChannel channel) throws Exception;
    
            private ReactorThread() throws IOException {
                selector = Selector.open();
            }
    
            volatile boolean running = false;
    
            @Override
            public void run() {
                // 轮询Selector事件
                while (running) {
                    try {
                        // 执行队列中的任务
                        Runnable task;
                        while ((task = taskQueue.poll()) != null) {
                            task.run();
                        }
                        selector.select(1000);
    
                        // 获取查询结果
                        Set<SelectionKey> selected = selector.selectedKeys();
                        // 遍历查询结果
                        Iterator<SelectionKey> iter = selected.iterator();
                        while (iter.hasNext()) {
                            // 被封装的查询结果
                            SelectionKey key = iter.next();
                            iter.remove();
                            int readyOps = key.readyOps();
                            // 关注 Read 和 Accept两个事件
                            if ((readyOps & (SelectionKey.OP_READ | SelectionKey.OP_ACCEPT)) != 0 || readyOps == 0) {
                                try {
                                    SelectableChannel channel = (SelectableChannel) key.attachment();
                                    channel.configureBlocking(false);
                                    handler(channel);
                                    if (!channel.isOpen()) {
                                        key.cancel(); // 如果关闭了,就取消这个KEY的订阅
                                    }
                                } catch (Exception ex) {
                                    key.cancel(); // 如果有异常,就取消这个KEY的订阅
                                }
                            }
                        }
                        selector.selectNow();
                    } catch (IOException e) {
                        e.printStackTrace();
                    }
                }
            }
    
            private SelectionKey register(SelectableChannel channel) throws Exception {
                // 为什么register要以任务提交的形式,让reactor线程去处理?
                // 因为线程在执行channel注册到selector的过程中,会和调用selector.select()方法的线程争用同一把锁
                // 而select()方法实在eventLoop中通过while循环调用的,争抢的可能性很高,为了让register能更快的执行,就放到同一个线程来处理
                FutureTask<SelectionKey> futureTask = new FutureTask<>(() -> channel.register(selector, 0, channel));
                taskQueue.add(futureTask);
                return futureTask.get();
            }
    
            private void doStart() {
                if (!running) {
                    running = true;
                    start();
                }
            }
        }
    
        private ServerSocketChannel serverSocketChannel;
        // 1、创建多个线程 - accept处理reactor线程 (accept线程)
        private ReactorThread[] mainReactorThreads = new ReactorThread[1];
        // 2、创建多个线程 - io处理reactor线程  (I/O线程)
        private ReactorThread[] subReactorThreads = new ReactorThread[8];
    
        /**
         * 初始化线程组
         */
        private void newGroup() throws IOException {
            // 创建IO线程,负责处理客户端连接以后socketChannel的IO读写
            for (int i = 0; i < subReactorThreads.length; i++) {
                subReactorThreads[i] = new ReactorThread() {
                    @Override
                    public void handler(SelectableChannel channel) throws IOException {
                        // work线程只负责处理IO处理,不处理accept事件
                        SocketChannel ch = (SocketChannel) channel;
                        ByteBuffer requestBuffer = ByteBuffer.allocate(1024);
                        while (ch.isOpen() && ch.read(requestBuffer) != -1) {
                            // 长连接情况下,需要手动判断数据有没有读取结束 (此处做一个简单的判断: 超过0字节就认为请求结束了)
                            if (requestBuffer.position() > 0) break;
                        }
                        if (requestBuffer.position() == 0) return; // 如果没数据了, 则不继续后面的处理
                        requestBuffer.flip();
                        byte[] content = new byte[requestBuffer.limit()];
                        requestBuffer.get(content);
                        System.out.println(new String(content));
                        System.out.println(Thread.currentThread().getName() + "收到数据,来自:" + ch.getRemoteAddress());
    
                        // TODO 业务操作 数据库、接口...
                        workPool.submit(() -> {
                        });
    
                        // 响应结果 200
                        String response = "HTTP/1.1 200 OK
    " +
                                "Content-Length: 11
    
    " +
                                "Hello World";
                        ByteBuffer buffer = ByteBuffer.wrap(response.getBytes());
                        while (buffer.hasRemaining()) {
                            ch.write(buffer);
                        }
                    }
                };
            }
    
            // 创建mainReactor线程, 只负责处理serverSocketChannel
            for (int i = 0; i < mainReactorThreads.length; i++) {
                mainReactorThreads[i] = new ReactorThread() {
                    AtomicInteger incr = new AtomicInteger(0);
    
                    @Override
                    public void handler(SelectableChannel channel) throws Exception {
                        // 只做请求分发,不做具体的数据读取
                        ServerSocketChannel ch = (ServerSocketChannel) channel;
                        SocketChannel socketChannel = ch.accept();
                        socketChannel.configureBlocking(false);
                        // 收到连接建立的通知之后,分发给I/O线程继续去读取数据
                        int index = incr.getAndIncrement() % subReactorThreads.length;
                        ReactorThread workEventLoop = subReactorThreads[index];
                        workEventLoop.doStart();
                        SelectionKey selectionKey = workEventLoop.register(socketChannel);
                        selectionKey.interestOps(SelectionKey.OP_READ);
                        System.out.println(Thread.currentThread().getName() + "收到新连接 : " + socketChannel.getRemoteAddress());
                    }
                };
            }
    
    
        }
    
        /**
         * 初始化channel,并且绑定一个eventLoop线程
         *
         * @throws IOException IO异常
         */
        private void initAndRegister() throws Exception {
            // 1、 创建ServerSocketChannel
            serverSocketChannel = ServerSocketChannel.open();
            serverSocketChannel.configureBlocking(false);
            // 2、 将serverSocketChannel注册到selector
            int index = new Random().nextInt(mainReactorThreads.length);
            mainReactorThreads[index].doStart();
            SelectionKey selectionKey = mainReactorThreads[index].register(serverSocketChannel);
            selectionKey.interestOps(SelectionKey.OP_ACCEPT);
        }
    
        /**
         * 绑定端口
         *
         * @throws IOException IO异常
         */
        private void bind() throws IOException {
            //  1、 正式绑定端口,对外服务
            serverSocketChannel.bind(new InetSocketAddress(8080));
            System.out.println("启动完成,端口8080");
        }
    
        public static void main(String[] args) throws Exception {
            NioServer3 nioServer3 = new NioServer3();
            nioServer3.newGroup(); // 1、 创建main和sub两组线程
            nioServer3.initAndRegister(); // 2、 创建serverSocketChannel,注册到mainReactor线程上的selector上
            nioServer3.bind(); // 3、 为serverSocketChannel绑定端口
        }
    
    
    
    }
  • 相关阅读:
    Android 常见工具类封装
    Android Logcat 封装类
    Android 四大组件之 " Activity "
    "浅谈Android"第一篇:Android系统简介
    罗列的书单
    关于多层架构一些思考
    LeetCode 330. Patching Array
    LeetCode 315. Count of Smaller Numbers After Self(线段树,树状数组)
    LeetCode 316. Remove Duplicate Letters(贪心)
    LeetCode 327. Count of Range Sum(线段树)
  • 原文地址:https://www.cnblogs.com/hhhshct/p/11763195.html
Copyright © 2011-2022 走看看