zoukankan      html  css  js  c++  java
  • NIO简介以及三大组件(BufferChannelSelector)基本使用

    1. 简介

    1. NIO全称Non-Blocking IO,是指JDK提供的新API。从JDK1.4开始,Java提供了一系列改进的输入/输出的新特性,被称为NIO(new IO),是同步非阻塞的。

    2. NIO的类被放在java.nio以及其子包下,并且对java.io包的很多类进行改造。

    3. NIO有三大核心部分:Channel(通道)、Buffer(缓冲区)、Selector(选择器)。Selector可以选择一个通道Channel,通道和Buffer缓冲区交互,客户端和Buffer缓冲区交互。

    4. NIO是面向缓冲区的,或者是面向块编程的。数据读到一个它稍后处理的缓冲区,需要时可在缓冲区中前后移动,这就增加了处理过程的灵活性,使用它可以提供非阻塞式的高伸缩性网络。

    5. JavaNIO的非阻塞模式,使一个线程从某通道发送或者请求数据,但是它仅能得到目前可用的数据,如果目前没有可用数据时,就什么都不会读取,而不是保持线程阻塞。所以到数据可读时,线程可以做其他事情。非阻塞写也是如此,一个线程请求写入一些数据到某通道,但不需要等待它完全写入,这个线程就可以做其他的事情。

    6. 通俗的来说,NIO 是可以做到一个线程来处理多个操作。

    7. HTTP2.0 使用了多路复用的技术,做到同一个连接并发处理多个请求,而且并发请求的数量比HTTP1.1 大了好几个数量级。

    2. BIO和NIO的关系

    1. BIO以流的方式处理数据,而NIO以块的方式处理数据,块I/O 的效率比流I/O 高很多

    2. BIO是阻塞的,而NIO则是非阻塞的

    3. BIO基于字节流和字符流进行操作,而NIO基于Channel(通道)和Buffer(缓冲区)进行操作, 数据总是从通道读取到缓冲区中,或者从缓冲区写入通道中。Selector(选择器) 用于监听多个通道的事件(比如连接请求、数据到达等),因此使用单个线程就可以监听到多个客户端通道。

    3. 三大组件的关系

    1》一个线程对应一个Selector,一个Selector 对应多个channel

    2》每个channel都对应一个buffer

    3》程序切换到哪个channel是由事件决定的,Event是一个重要的概念;Selector会根据不同的事件在不同的channel切换

    4》Buffer就是一个内存块,底层是有一个数组

    5》数据的读写是通过Buffer,这个和BIO是有区别的,BIO流要么是输入流、要么是输出流,不能双向,而NIO的BUFFER是双向的,可读可写,需要使用flip() 方法切换。

    6》channel 也是双向的,可以返回底层操作系统的操作情况,比如Linux底层的操作系统通道就是双向的。

    4. Buffer的机制及其子类

    Buffer 有四个重要的属性:

        // Invariants: mark <= position <= limit <= capacity
        private int mark = -1;
        private int position = 0;
        private int limit;
        private int capacity;

    capacity 表示容量,也就是最多存放的数量

    limit 表示缓冲区的当前终点,不能对缓冲区超过极限的位置进行读写操作。且极限是可以修改的。

    position 下一个要读或写的元素的索引

    mark 标记

    类继承关系如下:

    8大基本数据类型,除了bool都有对应的buffer。最常用的还是ByteBuffer。数据读取时根据数据类型放入到对应的Buffer。且每个子Buffer又有其子类,而且每个Buffer有存放数据的数组,比如IntBuffer如下:

        final int[] hb;                  // Non-null only for heap buffers
        final int offset;
        boolean isReadOnly;                 // Valid only for heap buffers

    测试如下:

    (1) 测试简单使用

    package cn.qlq;
    
    import java.nio.IntBuffer;
    
    public class BufferTestCase {
        
        public static void main(String[] args) {
            IntBuffer buffer = IntBuffer.allocate(5);
            // 存放数据
            for (int i = 0; i < buffer.capacity(); i++) {
                buffer.put(i * 2);
            }
            
            // buffer 读写切换
            buffer.flip();
            
            // 读取数据
            while (buffer.hasRemaining()) {
                System.out.println(buffer.get());
            }
        }
    
    }

    结果:

    0
    2
    4
    6
    8

     查看flip 的源码实际上就是修改limit 为position 的位置,并且将position 置为0。

        public final Buffer flip() {
            limit = position;
            position = 0;
            mark = -1;
            return this;
        }

    (2) 测试修改limit 和 position

        public static void main(String[] args) {
            IntBuffer buffer = IntBuffer.allocate(5);
            // 存放数据
            for (int i = 0; i < buffer.capacity(); i++) {
                buffer.put(i * 2);
            }
            
            // buffer 读写切换
            buffer.flip();
            
            // 修改limit 和 position
            buffer.limit(3);
            buffer.position(1);
            
            // 读取数据
            while (buffer.hasRemaining()) {
                System.out.println(buffer.get());
            }
        }

    结果:

    2
    4

    (3) 其他方法介绍

    buffer.clear(); // 清除缓冲区,各个标记恢复到初始标记,但是不清楚数据

    put(int index, byte b) 向指定位置插入数据

    get(int index) 读取指定位置的数据

    测试:

        public static void main(String[] args) {
            ByteBuffer buffer = ByteBuffer.allocate(5);
            // 存放数据
            buffer.put(2, (byte) 2);
    
            // buffer 读写切换
            buffer.flip(); // 反转缓冲区
    
            // 修改limit 和 position
            buffer.clear(); // 清除缓冲区,各个标记恢复到初始标记,但是不清楚数据
            System.out.println(buffer.get(2));
        }

    结果:

    2

    (4) buffer 支持类型化的put和get,put进去什么类型的数据,get出来时就应该使用相应的数据类型取出,否则会报错:BufferUnderflowException, 当然如果可以自动类型转换不会报错

            ByteBuffer buffer = ByteBuffer.allocate(512);
            buffer.putInt(1);
            buffer.putLong(2L);
            buffer.putShort((short)2);
            buffer.putChar('2');
    
            // 切换读写
            buffer.flip();
    
            System.out.println(buffer.getInt());
            System.out.println(buffer.getLong());
            System.out.println(buffer.getShort());
            System.out.println(buffer.getLong());

    结果:

    1
    2
    2
    Exception in thread "main" java.nio.BufferUnderflowException
        at java.nio.Buffer.nextGetIndex(Buffer.java:506)
        at java.nio.HeapByteBuffer.getLong(HeapByteBuffer.java:412)
        at nio.ChannelTest.bufferTest(ChannelTest.java:29)
        at nio.ChannelTest.main(ChannelTest.java:13)

    (5) buffer 可以设置为只读,写入时会报错

            ByteBuffer buffer = ByteBuffer.allocate(512);
            System.out.println(buffer.getClass());
            System.out.println(buffer.isReadOnly());
    
            ByteBuffer byteBuffer = buffer.asReadOnlyBuffer();
            System.out.println(buffer.getClass());
            System.out.println(byteBuffer.isReadOnly());
    
            byteBuffer.putLong(2L);

    结果:

    class java.nio.HeapByteBuffer
    false
    class java.nio.HeapByteBuffer
    true
    Exception in thread "main" java.nio.ReadOnlyBufferException
        at java.nio.HeapByteBufferR.putLong(HeapByteBufferR.java:426)
        at nio.ChannelTest.bufferTest(ChannelTest.java:25)
        at nio.ChannelTest.main(ChannelTest.java:13)

    (6) NIO还提供了MappedByteBuffer,可以让文件直接在内存(堆外内存)中直接修改,而如何同步到文件中由NIO来完成

    类图如下:

    可以看到最底层的是一个只读的Buffer

    1》测试如下:

            // 以读写模式打开文件1.txt
            RandomAccessFile file = new RandomAccessFile("1.txt", "rw");
    
            FileChannel channel = file.getChannel();
    
            /**
             * 映射成MappedByteBuffer(读写模式,从0开始,最多映射五个,也就是最多可以在内存中修改五个)
             *
             */
            MappedByteBuffer mappedByteBuffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, 5);
            mappedByteBuffer.putChar(3, '3');
    
            file.close();
            System.out.println("修改成功" + mappedByteBuffer.getClass());

    结果:文件会更新成功,最后控制台打印如下:

    修改成功class java.nio.DirectByteBuffer

    2》 测试二: 修改索引为5的时候报异常

            // 以读写模式打开文件1.txt
            RandomAccessFile file = new RandomAccessFile("1.txt", "rw");
    
            FileChannel channel = file.getChannel();
    
            /**
             * 映射成MappedByteBuffer(读写模式,从0开始,最多映射五个,也就是最多可以在内存中修改五个=也就是0-4)
             *
             */
            MappedByteBuffer mappedByteBuffer = channel.map(FileChannel.MapMode.READ_WRITE, 0, 5);
            mappedByteBuffer.putChar(3, '3');
            mappedByteBuffer.putChar(5, '3');
    
            file.close();
            System.out.println("修改成功" + mappedByteBuffer.getClass());

    结果:

    Exception in thread "main" java.lang.IndexOutOfBoundsException
        at java.nio.Buffer.checkIndex(Buffer.java:546)
        at java.nio.DirectByteBuffer.putChar(DirectByteBuffer.java:533)
        at nio.ChannelTest.bufferTest(ChannelTest.java:30)
        at nio.ChannelTest.main(ChannelTest.java:14)

    (7) buffer也可以作为数组使用来完成读写操作,即Scattering(分散)和Gathering(聚集)

            // 使用ServerSocketChannel 和 SocketChannel
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
            // 绑定端口
            InetSocketAddress inetSocketAddress = new InetSocketAddress(7000);
            serverSocketChannel.socket().bind(inetSocketAddress);
            ByteBuffer[] byteBuffers = new ByteBuffer[2];
            byteBuffers[0] = ByteBuffer.allocate(5);
            byteBuffers[1] = ByteBuffer.allocate(3);
    
            SocketChannel socketChannel = serverSocketChannel.accept();
            int msgLength = 8;
            while (true) {
                int readed = 0;
                while (readed < msgLength) {
                    long l = socketChannel.read(byteBuffers);
                    readed += l;
                    System.out.println("readed: " + readed);
    
                    Arrays.asList(byteBuffers).stream()
                            .map(buffer -> "position = " + buffer.position() + ", limit = " + buffer.limit())
                            .forEach(System.out::println);
                    Arrays.asList(byteBuffers).stream()
                            .forEach(buffer -> System.out.println("data: " + new String(buffer.array())));
                }
    
                // 数据显示到客户端
                Arrays.asList(byteBuffers).stream().forEach(buffer -> buffer.flip());
                long byteWrite = 0;
                while (byteWrite < msgLength) {
                    long l = socketChannel.write(byteBuffers);
                    byteWrite += l;
                }
    
                Arrays.asList(byteBuffers).stream().forEach(buffer -> buffer.clear());
    
                System.out.println("byteWrite = " + byteWrite + ",  readed = " + readed + " , msgLength = " + msgLength);
            }

    测试:

    1》telnet连接

    telnet 127.0.0.1 7000

    2》Ctrl +] 

    3》发送命令

    Microsoft Telnet> send hello123
    发送字符串 hello123

    4》控制台日志如下

    readed: 8
    position = 5, limit = 5
    position = 3, limit = 3
    data: hello
    data: 123
    byteWrite = 8, readed = 8 , msgLength = 8

    5. Channel通道

    1. 简介

    1. NIO channel 通道类似于流,但是有区别:

    (1) 通道可以同时读或者写,而流只能读或者只能写

    (2)通道可以实现异步读写数据

    (3)通道可以从缓冲区读数据,也可以写数据到缓冲区

    2. Channel 在NIO中是一个接口,下面有很多子接口和主要实现。主要实现有:

    FileChannel 注意用于文件的数据读写,DatagramChannel 用于UDP的数据读写,ServerSocketChannel和 SocketChannel 用于TCP的数据读写。

    2. FileChannel 简单使用

    1. 简单的使用FileChannel将字符串写入到文件:

        private static void fileChannelTest() throws Exception {
            String string = "hello,中国!";
            FileOutputStream fileOutputStream = new FileOutputStream("F:/test.txt");
            
            // 通过FileOutputStream 获取 FileChannelImpl
            FileChannel channel = fileOutputStream.getChannel();
            
            // 创建一个缓冲区
            ByteBuffer byteBuffer = ByteBuffer.allocate(1024);
            // 数据放入buffer
            byteBuffer.put(string.getBytes());
            
            // 对buffer进行flip切换读写
            byteBuffer.flip();
            channel.write(byteBuffer);
            fileOutputStream.close();
        }

    put完之后debug查看buffer信息:(一个汉字占三个字节,7个英文字母加2个汉字所以是13bytes)

    2.  使用FileChannel 读取文件:

    package cn.qlq;
    
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileNotFoundException;
    import java.io.FileOutputStream;
    import java.nio.ByteBuffer;
    import java.nio.channels.FileChannel;
    
    public class ChannelTest {
    
        public static void main(String[] args) throws Exception {
            fileChannelTest();
        }
    
        private static void fileChannelTest() throws Exception {
            File file = new File("F:/test.txt");
            FileInputStream inputStream = new FileInputStream(file);
            
            // 通过FileInputStream 获取 FileChannelImpl
            FileChannel channel = inputStream.getChannel();
            
            // 创建一个缓冲区
            ByteBuffer byteBuffer = ByteBuffer.allocate((int)file.length());
            // 数据读到缓冲中
            channel.read(byteBuffer);
            
            System.out.println(new String(byteBuffer.array()));
            inputStream.close();
        }
    }

    结果:

    hello,中国!

     3. 使用FileChannel读取一个文件并写入另一个文件:

        @SneakyThrows
        private static void fileChannelTest() {
            FileInputStream inputStream = new FileInputStream("1.txt");
            FileChannel channel1 = inputStream.getChannel();
            FileOutputStream outputStream = new FileOutputStream("2.txt");
            FileChannel channel2 = outputStream.getChannel();
    
            // 创建一个缓冲区
            ByteBuffer byteBuffer = ByteBuffer.allocate(512);
    
            while (true) {
                int read = channel1.read(byteBuffer);
                if (read == -1) {
                    break;
                }
    
                byteBuffer.flip();
                channel2.write(byteBuffer);
    
                //关键操作. 清空Buffer, 将关键的属性重置. 如果没有这一步会一直read到0
                byteBuffer.clear();
                System.out.println(read);
            }
    
            inputStream.close();
            outputStream.close();
        }

    4 . FileChannel 实现文件拷贝

        @SneakyThrows
        private static void fileChannelTest() {
            FileInputStream inputStream = new FileInputStream("1.txt");
            FileChannel channel1 = inputStream.getChannel();
            FileOutputStream outputStream = new FileOutputStream("3.txt");
            FileChannel channel2 = outputStream.getChannel();
    
            // channel拷贝,从输入流的channel拷贝到输出流的channel
            channel2.transferFrom(channel1, 0, channel1.size());
    
            inputStream.close();
            outputStream.close();
        }

       经过上面四个例子明白。每个Stream都包含一个Channel,向Channel写入或者读取会操作到对应的Stream。

    6. Selector 选择器

    1. Java的NIO,用非阻塞的IO方式,可以用一个线程,处理多个客户端连接,就会使用到Selector(选择器)

    2. Selector 能够检测多个注册的通道上是否有事件发生(多个Channel可以以事件的方式注册到Selector),如果有事件发生便获取事件然后针对每个事件进行相应的处理,这样就可以只用一个单线程去管理多个通道,也就是管理多个连接和请求

    3. 只有在连接真正有读写事件发生时,才会进行读写,就大大地减少了系统开销,并且不必为每个连接都创建一个线程,不用去维护多个线程

    4. 避免了多线程之间的上下文且护眼导致的开销

    特点:

    (1) Netty的IO线程NioEventLoop聚合了Selector(选择器,也叫多路复用器),可以同时并发处理成百上千个客户端连接

    (2) 当线程从某客户端Socket 通道进行读写数据时,在线程没有数据可用时,该线程可以执行其他任务

    (3) 线程通道将非阻塞IO的空闲时间用于在其他通道上执行IO操作,所以单独的线程可以管理多个输入和输出通道

    (4) 由于读写操作都是非阻塞的,这就可以充分提升IO线程的运行效率,避免由于频繁IO阻塞导致的线程挂起

    (5) 一个IO线程可以并发处理N个客户连接和读写操作,这从根本上解决了传统同步阻塞IO一连接一线程模型,架构的性能、弹性伸缩能力和可靠性都得到了极大的提升

    Selector 源码如下:

    public abstract class Selector implements Closeable {
        protected Selector() {
        }
    
        public static Selector open() throws IOException {
            return SelectorProvider.provider().openSelector();
        }
    
        public abstract boolean isOpen();
    
        public abstract SelectorProvider provider();
    
        public abstract Set<SelectionKey> keys();
    
        public abstract Set<SelectionKey> selectedKeys();
    
        public abstract int selectNow() throws IOException;
    
        public abstract int select(long var1) throws IOException;
    
        public abstract int select() throws IOException;
    
        public abstract Selector wakeup();
    
        public abstract void close() throws IOException;
    }

    核心方法:

    open    得到一个选择器对象,也就是获取Selector的方法。

    select()    选择一个准备好的key集合,为IO操作做准备。这个是一个阻塞的方法,会阻塞到获取到数据。

    select(long var1)    监控所有注册的通道,当其中有IO操作可以进行时,将对应的SelectionKey加入到内部集合中并返回,参数用来设置超时时间,这个是非阻塞的。SelectionKey 是存在对象内部集合的一个元素,通过SelectionKey可以获取到Channel、注册的事件(读、写)等信息。

    selectNow()    也是非阻塞的,只是获取一下,如果有SelectionKey 就加入内部集合,没有就放弃。

    1. NIO 非阻塞 网络编程相关的(Selector、SelectionKey、ServerScoketChannel和SocketChannel)  关系

    1. 当客户端连接时,会通过ServerSocketChannel 得到 SocketChannel

    2. Selector 进行监听 select 方法, 返回有事件发生的通道的个数

    3. 将socketChannel注册到Selector上, register(Selector sel, int ops), 一个selector上可以注册多个SocketChannel。ops 代表的是注册的事件。

    java.nio.channels.SelectionKey#OP_READ 读事件

    java.nio.channels.SelectionKey#OP_WRITE 写事件

    java.nio.channels.SelectionKey#OP_CONNECT 连接建立成功事件

    java.nio.channels.SelectionKey#OP_ACCEPT 有一个新连接事件

    4. 注册后返回一个 SelectionKey, 会和该Selector 关联(集合)

    5. 进一步得到各个 SelectionKey (有事件发生)

    6. 在通过 SelectionKey 反向获取 SocketChannel , 方法 channel()

    7. 可以通过 得到的 channel , 完成业务处理

    2. 测试:

    (1) 服务器端源码

    package nio;
    
    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 NIOServer {
    
        public static void main(String[] args) throws Exception {
            // 创建ServerSocketChannel -> ServerSocket
            // Java NIO中的 ServerSocketChannel 是一个可以监听新进来的TCP连接的通道, 就像标准IO中的ServerSocket一样。ServerSocketChannel类在 java.nio.channels包中。
            // 通过调用 ServerSocketChannel.open() 方法来打开ServerSocketChannel.如:
            ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
    
            // 得到一个Selecor对象 (sun.nio.ch.WindowsSelectorImpl)
            Selector selector = Selector.open();
            //绑定一个端口6666, 在服务器端监听
            serverSocketChannel.socket().bind(new InetSocketAddress(6666));
            //设置为非阻塞
            serverSocketChannel.configureBlocking(false);
    
            //把 serverSocketChannel 注册到  selector 关心 事件为 OP_ACCEPT
            //SelectionKey中定义的4种事件
            //SelectionKey.OP_ACCEPT —— 接收连接进行事件,表示服务器监听到了客户连接,那么服务器可以接收这个连接了
            // SelectionKey.OP_CONNECT —— 连接就绪事件,表示客户与服务器的连接已经建立成功
            //SelectionKey.OP_READ  —— 读就绪事件,表示通道中已经有了可读的数据,可以执行读操作了(通道目前有数据,可以进行读操作了)
            //SelectionKey.OP_WRITE —— 写就绪事件,表示已经可以向通道写数据了(通道目前可以用于写操作)
            serverSocketChannel.register(selector, SelectionKey.OP_ACCEPT);
    
            System.out.println("注册后的selectionkey 数量=" + selector.keys().size()); // 1
    
            //循环等待客户端连接
            while (true) {
                //这里我们等待1秒,如果没有事件发生, 返回
                if (selector.select(1000) == 0) { //没有事件发生
    //                System.out.println("服务器等待了1秒,无连接");
                    continue;
                }
    
                //如果返回的>0, 就获取到相关的 selectionKey集合
                //1.如果返回的>0, 表示已经获取到关注的事件
                //2. selector.selectedKeys() 返回关注事件的集合
                //   通过 selectionKeys 反向获取通道
                Set<SelectionKey> selectionKeys = selector.selectedKeys();
                System.out.println("selectionKeys 数量 = " + selectionKeys.size());
    
                //hasNext() :该方法会判断集合对象是否还有下一个元素,如果已经是最后一个元素则返回false。
                //next():把迭代器的指向移到下一个位置,同时,该方法返回下一个元素的引用。
                //remove() 从迭代器指向的集合中移除迭代器返回的最后一个元素。
                //遍历 Set<SelectionKey>, 使用迭代器遍历
                Iterator<SelectionKey> keyIterator = selectionKeys.iterator();
    
                while (keyIterator.hasNext()) {
                    //获取到SelectionKey
                    SelectionKey key = keyIterator.next();
                    //根据key 对应的通道发生的事件做相应处理
                    if (key.isAcceptable()) { //如果是 OP_ACCEPT, 有新的客户端连接
                        //该该客户端生成一个 SocketChannel
                        SocketChannel socketChannel = serverSocketChannel.accept();
                        System.out.println("客户端连接成功 生成了一个 socketChannel " + socketChannel.hashCode());
                        //将  SocketChannel 设置为非阻塞
                        socketChannel.configureBlocking(false);
                        //将socketChannel 注册到selector, 关注事件为 OP_READ, 同时给socketChannel 关联一个Buffer
                        socketChannel.register(selector, SelectionKey.OP_READ, ByteBuffer.allocate(1024));
    
                        System.out.println("客户端连接后 ,注册的selectionkey 数量=" + selector.keys().size()); //2,3,4..
                    }
    
                    if (key.isReadable()) {  //发生 OP_READ
                        //通过key 反向获取到对应channel
                        SocketChannel channel = (SocketChannel) key.channel();
                        //获取到该channel关联的buffer
                        ByteBuffer buffer = (ByteBuffer) key.attachment();
                        channel.read(buffer);
                        System.out.println("from 客户端: " + new String(buffer.array()));
                    }
    
                    //手动从集合中移动当前的selectionKey, 防止重复操作
                    keyIterator.remove();
                }
            }
        }
    }

    (2) 客户端代码

    package nio;
    
    import java.net.InetSocketAddress;
    import java.nio.ByteBuffer;
    import java.nio.channels.SocketChannel;
    
    public class NIOClient {
    
        public static void main(String[] args) throws Exception {
            //得到一个网络通道
            SocketChannel socketChannel = SocketChannel.open();
            //设置非阻塞
            socketChannel.configureBlocking(false);
            //提供服务器端的ip 和 端口
            InetSocketAddress inetSocketAddress = new InetSocketAddress("127.0.0.1", 6666);
            //连接服务器
            if (!socketChannel.connect(inetSocketAddress)) {
                while (!socketChannel.finishConnect()) {
                    System.out.println("因为连接需要时间,客户端不会阻塞,可以做其它工作..");
                }
            }
    
            //如果连接成功,就发送数据
            String str = "hello, China ~ ";
            //Wraps a byte array into a buffer
            ByteBuffer buffer = ByteBuffer.wrap(str.getBytes());
            //发送数据,将 buffer 数据写入 channel
            socketChannel.write(buffer);
            System.in.read();
        }
    }

    结果:服务器端控制台

    selectionKeys 数量 = 1
    客户端连接成功 生成了一个 socketChannel 1722023916
    客户端连接后 ,注册的selectionkey 数量=2
    selectionKeys 数量 = 1
    from 客户端 hello, China~                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                   

    3. 小结

    1 SelectionKey:表示 Selector 和网络通道的注册关系, 共四种:

    java.nio.channels.SelectionKey#OP_READ 读事件

    java.nio.channels.SelectionKey#OP_WRITE 写事件

    java.nio.channels.SelectionKey#OP_CONNECT 连接建立成功事件

    java.nio.channels.SelectionKey#OP_ACCEPT 有一个新连接事件

    2.   SelectionKey 的关键方法

    public abstract class SelectionKey {
        public static final int OP_READ = 1;    // 读事件
        public static final int OP_WRITE = 4;    // 写事件
        public static final int OP_CONNECT = 8;    // 链接建立事件
        public static final int OP_ACCEPT = 16;    // 请求建立连接事件
        private volatile Object attachment = null;
        private static final AtomicReferenceFieldUpdater<SelectionKey, Object> attachmentUpdater = AtomicReferenceFieldUpdater.newUpdater(SelectionKey.class, Object.class, "attachment");
    
        protected SelectionKey() {
        }
    
        public abstract SelectableChannel channel();    // //得到与之关联的通道
    
        public abstract Selector selector();    //得到与之关联的 Selector 对象
    
        public abstract boolean isValid();
    
        public abstract void cancel();
    
        public abstract int interestOps();    // 
    
        public abstract SelectionKey interestOps(int var1);    ////设置或改变监听事件
    
        public abstract int readyOps();
    
        public final boolean isReadable() {    // 是否可读,也就是是否读事件
            return (this.readyOps() & 1) != 0;
        }
    
        public final boolean isWritable() {
            return (this.readyOps() & 4) != 0;
        }
    
        public final boolean isConnectable() {
            return (this.readyOps() & 8) != 0;
        }
    
        public final boolean isAcceptable() {
            return (this.readyOps() & 16) != 0;
        }
    
        public final Object attach(Object var1) {
            return attachmentUpdater.getAndSet(this, var1);
        }
    
        public final Object attachment() {    // //得到与之关联的共享数据
            return this.attachment;
        }
    }

    3. ServerSocketChannel 关键方法-核心是连接过来之后建立连接

    public abstract class ServerSocketChannel extends AbstractSelectableChannel implements NetworkChannel {
        protected ServerSocketChannel(SelectorProvider var1) {
            super(var1);
        }
    
        public static ServerSocketChannel open() throws IOException {    // 得到一个 ServerSocketChannel 通道
            return SelectorProvider.provider().openServerSocketChannel();
        }
    
        public final int validOps() {
            return 16;
        }
    
        public final ServerSocketChannel bind(SocketAddress var1) throws IOException {    // 设置服务器端端口号
            return this.bind(var1, 0);
        }
    
        public abstract ServerSocketChannel bind(SocketAddress var1, int var2) throws IOException;
    
        public abstract <T> ServerSocketChannel setOption(SocketOption<T> var1, T var2) throws IOException;
    
        public abstract ServerSocket socket();
    
        public abstract SocketChannel accept() throws IOException;    // 接受一个连接,返回代表这个连接的通道对象
    
        public abstract SocketAddress getLocalAddress() throws IOException;
    }

    还有从父类继承的一些重要方法:

    java.nio.channels.spi.AbstractSelectableChannel#configureBlocking    // 设置阻塞或非阻塞模式,取值 false 表示采用非阻塞模式
    java.nio.channels.SelectableChannel#register(java.nio.channels.Selector, int)    // 注册一个选择器并设置监听事件

    4. SocketChannel 网络 IO 通道,具体负责进行读写操作。NIO 把缓冲区的数据写入通道,或者把通道里的数据读到缓冲区。

    public abstract class SocketChannel extends AbstractSelectableChannel implements ByteChannel, ScatteringByteChannel, GatheringByteChannel, NetworkChannel {
        protected SocketChannel(SelectorProvider var1) {
            super(var1);
        }
    
        public static SocketChannel open() throws IOException {    // 打开一个SocketChannel
            return SelectorProvider.provider().openSocketChannel();
        }
    
        public static SocketChannel open(SocketAddress var0) throws IOException {
            SocketChannel var1 = open();
    
            try {
                var1.connect(var0);
            } catch (Throwable var5) {
                try {
                    var1.close();
                } catch (Throwable var4) {
                    var5.addSuppressed(var4);
                }
    
                throw var5;
            }
    
            assert var1.isConnected();
    
            return var1;
        }
    
        public final int validOps() {
            return 13;
        }
    
        public abstract SocketChannel bind(SocketAddress var1) throws IOException;
    
        public abstract <T> SocketChannel setOption(SocketOption<T> var1, T var2) throws IOException;
    
        public abstract SocketChannel shutdownInput() throws IOException;
    
        public abstract SocketChannel shutdownOutput() throws IOException;
    
        public abstract Socket socket();
    
        public abstract boolean isConnected();
    
        public abstract boolean isConnectionPending();
    
        public abstract boolean connect(SocketAddress var1) throws IOException;    // //连接服务器
    
        public abstract boolean finishConnect() throws IOException;    // 如果上面的connect方法连接失败,接下来就要通过该方法完成连接操作
    
        public abstract SocketAddress getRemoteAddress() throws IOException;
    
        public abstract int read(ByteBuffer var1) throws IOException;    //从通道里读数据
    
        public abstract long read(ByteBuffer[] var1, int var2, int var3) throws IOException;
    
        public final long read(ByteBuffer[] var1) throws IOException {
            return this.read(var1, 0, var1.length);
        }
    
        public abstract int write(ByteBuffer var1) throws IOException;    //往通道里写数据
    
        public abstract long write(ByteBuffer[] var1, int var2, int var3) throws IOException;
    
        public final long write(ByteBuffer[] var1) throws IOException {
            return this.write(var1, 0, var1.length);
        }
    
        public abstract SocketAddress getLocalAddress() throws IOException;
    }

    还有从父类继承的一些重要方法:

    java.nio.channels.spi.AbstractSelectableChannel#configureBlocking    // 设置阻塞或非阻塞模式,取值 false 表示采用非阻塞模式
    java.nio.channels.SelectableChannel#register(java.nio.channels.Selector, int)    // 注册一个选择器并设置监听事件
    【当你用心写完每一篇博客之后,你会发现它比你用代码实现功能更有成就感!】
  • 相关阅读:
    关于 log4j.additivity
    JDK8新特性:使用Optional:解决NPE问题的更干净的写法
    异常处理和日志输出使用小结
    搭建DNS服务器
    git 使用技巧
    mysql
    linux学习记录
    nginx解析
    node npm pm2命令简析
    jenkins使用简析
  • 原文地址:https://www.cnblogs.com/qlqwjy/p/14433524.html
Copyright © 2011-2022 走看看