Netty学习日记一:三大组件

1 篇文章 0 订阅
1 篇文章 0 订阅

Channel & Buffer

channel 有一点类似于 stream,它就是读写数据的双向通道,可以从 channel 将数据读入 buffer,也可以将 buffer 的数据写入 channel,而之前的 stream 要么是输入,要么是输出,channel 比 stream 更为底层

channel
buffer

常见的 Channel

  • FileChannel
  • DatagramChannel
  • SocketChannel
  • ServerSocketChannel

常见的 buffer

buffer 则用来缓冲读写数据

  • ByteBuffer
    • MappedByteBuffer
    • DirectByteBuffer
    • HeapByteBuffer
  • ShortBuffer
  • IntBuffer
  • LongBuffer
  • FloatBuffer
  • DoubleBuffer
  • CharBuffer

FileChannel的用法

文件通道FileChannel是用于读取,写入,文件的通道。FileChannel只能被InputStream、OutputStream、RandomAccessFile创建。FileChannel通过字节流创建,那么操作的缓冲区肯定就是字节缓冲区。

import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.nio.ByteBuffer;
import java.nio.channels.FileChannel;

/**
 * @author: 随风飘的云
 * @describe:FileChannel的用法
 * @date 2022/08/04 22:48
 */
public class FileChannelTest {
    // 使用通道读文件
    public void readData(File file){
        FileInputStream inputStream = null;
        try {
            inputStream = new FileInputStream(file);
            // 获取文件通道
            FileChannel channel = inputStream.getChannel();
            // 创建缓冲区
            ByteBuffer buffer = ByteBuffer.allocate(1024);
            // 从通道读取数据到缓冲区
            channel.read(buffer);
            // 反转缓冲区(limit设置为position,position设置为0,mark设置为-1)
            buffer.flip();
            // 就是判断position和limit之间是否有元素
            while (buffer.hasRemaining()){
                // 按照字节的格式获取数据
                System.out.print((char) buffer.get());
            }
            buffer.clear();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            try {
                if(inputStream != null){
                    inputStream.close();
                }
            }catch (Exception e){
                e.printStackTrace();
            }
        }
    }

    public void writeData(File file, String data){
        FileOutputStream outputStream = null;
        try {
            outputStream = new FileOutputStream(file);
            // 获取文件通道
            FileChannel channel = outputStream.getChannel();
            // 创建缓冲区
            ByteBuffer buffer = ByteBuffer.allocate(1024);
            // 将数据装入缓冲区
            buffer.put(data.getBytes());
            // 反转缓冲区(limit设置为position,position设置为0,mark设置为-1)
            buffer.flip();
            // 从通道读取数据到缓冲区
            channel.write(buffer);
            // 将缓冲区中的数据写入到通道
            channel.write(buffer);
            // 写完将缓冲区还原(position设置为0,limit设置为capacity,mark设置为-1)
            buffer.clear();
        }catch (Exception e){
            e.printStackTrace();
        }finally {
            try {
                if(outputStream != null){
                    outputStream.close();
                }
            }catch (Exception e){
                e.printStackTrace();
            }
        }
    }

    public static void main(String[] args) {
        FileChannelTest channelTest = new FileChannelTest();
        File file = new File("xxx.xxx  文件");
        channelTest.readData(file);
        channelTest.writeData(file, "1234567890");
    }
}

结果: 获取xxx文件的内容并且把1234567890写进该文件中。

FileChannel的源码解析

包位置:

package java.nio.channels;

继承与实现关系

public abstract class FileChannel  
    extends AbstractInterruptibleChannel  
    implements SeekableByteChannel, GatheringByteChannel, ScatteringByteChannel

构造器

  /** 
     * 初始化一个无参构造器. 
     */  
    protected FileChannel() { }

具体方法:

//打开或创建一个文件,返回一个文件通道来访问文件  
    public static FileChannel open(Path path,  
                                   Set<? extends OpenOption> options,  
                                   FileAttribute<?>... attrs)  
        throws IOException  
    {  
        FileSystemProvider provider = path.getFileSystem().provider();  
        return provider.newFileChannel(path, options, attrs);  
    }  
  
    private static final FileAttribute<?>[] NO_ATTRIBUTES = new FileAttribute[0];  
  
    //打开或创建一个文件,返回一个文件通道来访问文件  
    public static FileChannel open(Path path, OpenOption... options)  
        throws IOException  
    {  
        Set<OpenOption> set = new HashSet<OpenOption>(options.length);  
        Collections.addAll(set, options);  
        return open(path, set, NO_ATTRIBUTES);  
    }  
  
    //从这个通道读入一个字节序列到给定的缓冲区  
    public abstract int read(ByteBuffer dst) throws IOException;  
  
    //从这个通道读入指定开始位置和长度的字节序列到给定的缓冲区  
    public abstract long read(ByteBuffer[] dsts, int offset, int length)  
        throws IOException;  
  
    /** 
     * 从这个通道读入一个字节序列到给定的缓冲区 
     */  
    public final long read(ByteBuffer[] dsts) throws IOException {  
        return read(dsts, 0, dsts.length);  
    }  
  
    /** 
     * 从给定的缓冲区写入字节序列到这个通道 
     */  
    public abstract int write(ByteBuffer src) throws IOException;  
  
    /** 
     * 从给定缓冲区的子序列向该信道写入字节序列 
     */  
    public abstract long write(ByteBuffer[] srcs, int offset, int length)  
        throws IOException;  
  
    /** 
     * 从给定的缓冲区写入字节序列到这个通道 
     */  
    public final long write(ByteBuffer[] srcs) throws IOException {  
        return write(srcs, 0, srcs.length);  
    }  
  
    /** 
     * 返回通道读写缓冲区中的开始位置 
     */  
    public abstract long position() throws IOException;  
  
    /** 
     * 设置通道读写缓冲区中的开始位置 
     */  
    public abstract FileChannel position(long newPosition) throws IOException;  
  
    /** 
     * 返回此通道文件的当前大小 
     */  
    public abstract long size() throws IOException;  
  
    /** 
     * 通过指定的参数size来截取通道的大小 
     */  
    public abstract FileChannel truncate(long size) throws IOException;  
  
    /** 
     * 强制将通道中的更新文件写入到存储设备(磁盘等)中 
     */  
    public abstract void force(boolean metaData) throws IOException;  
  
    /** 
     * 将当前通道中的文件写入到可写字节通道中 
     * position就是开始写的位置,long就是写的长度 
     */  
    public abstract long transferTo(long position, long count,  
                                    WritableByteChannel target)  
        throws IOException;  
  
    /** 
     * 将当前通道中的文件写入可读字节通道中 
     * position就是开始写的位置,long就是写的长度 
     */  
    public abstract long transferFrom(ReadableByteChannel src,  
                                      long position, long count)  
        throws IOException;  
  
    /** 
     * 从通道中读取一系列字节到给定的缓冲区中 
     * 从指定的读取开始位置position处读取 
     */  
    public abstract int read(ByteBuffer dst, long position) throws IOException;  
  
    /** 
     * 从给定的缓冲区写入字节序列到这个通道 
     * 从指定的读取开始位置position处开始写 
     */  
    public abstract int write(ByteBuffer src, long position) throws IOException;  
  
  
    // -- Memory-mapped buffers --  
  
    /** 
     * 一个文件映射模式类型安全枚举 
     */  
    public static class MapMode {  
  
        //只读映射模型  
        public static final MapMode READ_ONLY  
            = new MapMode("READ_ONLY");  
  
        //读写映射模型  
        public static final MapMode READ_WRITE  
            = new MapMode("READ_WRITE");  
  
        /** 
         * 私有模式(复制在写)映射 
         */  
        public static final MapMode PRIVATE  
            = new MapMode("PRIVATE");  
  
        private final String name;  
  
        private MapMode(String name) {  
            this.name = name;  
        }  
    }  
  
    /** 
     * 将该通道文件的一个区域直接映射到内存中 
     */  
    public abstract MappedByteBuffer map(MapMode mode,  
                                         long position, long size)  
        throws IOException;  
  
    /** 
     * 获取当前通道文件的给定区域上的锁 
     * 区域就是从position处开始,size长度  
     * shared为true代表获取共享锁,false代表获取独占锁 
     */  
    public abstract FileLock lock(long position, long size, boolean shared)  
        throws IOException;  
  
    /** 
     * 获取当前通道文件上的独占锁 
     */  
    public final FileLock lock() throws IOException {  
        return lock(0L, Long.MAX_VALUE, false);  
    }  
  
    /** 
     * 尝试获取给定的通道文件区域上的锁 
     * 区域就是从position处开始,size长度  
     * shared为true代表获取共享锁,false代表获取独占锁 
     */  
    public abstract FileLock tryLock(long position, long size, boolean shared)  
        throws IOException;  
  
    /** 
     * 尝试获取当前通道文件上的独占锁 
     */  
    public final FileLock tryLock() throws IOException {  
        return tryLock(0L, Long.MAX_VALUE, false);  
    }  

Selector

Selector的作用是Java NIO中管理一组多路复用的SelectableChannel对象,并能够识别通道是否为诸如读写事件做好准备的组件selector。单从字面意思不好理解,需要结合服务器的设计演化来理解它的用途。

多线程版设计

多线程版
socket1
thread
socket2
thread
socket3
thread

⚠️多线程版缺点

  • 内存占用高
  • 线程上下文切换成本高
  • 只适合连接数少的场景

线程池版设计

线程池版
socket1
thread
socket2
thread
socket3
socket4

⚠️线程池版缺点

  • 阻塞模式下,线程仅能处理一个 socket 连接
  • 仅适合短连接场景

selector 版设计

selector 的作用就是配合一个线程来管理多个 channel,获取这些 channel 上发生的事件,这些 channel 工作在非阻塞模式下,不会让线程吊死在一个 channel 上。适合连接数特别多,但流量低的场景(low traffic)

selector 版
selector
thread
channel
channel
channel

调用 selector 的 select() 会阻塞直到 channel 发生了读写就绪事件,这些事件发生,select 方法就会返回这些事件交给 thread 来处理

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值