NIO与零拷贝

作者: 贪挽懒月 | 来源:发表于2020-07-18 17:37 被阅读0次

一、是什么

先来看如下一段代码:

File file = new File("test.txt");
RandomAccessFile raf = new RandomAccessFile(file, "rw");
        
byte[] arr = new byte[(int)file.length()];
raf.read(arr);
        
Socket socket = new ServerSocket(8888).accept();
socket.getOutputStream().write(arr);

这段代码就是读取一个文件,然后再把它写出去,看起来就几行代码,其实涉及到多次拷贝,其流程如下:

  • 读取需要拷贝的数据,这个过程有两个步骤:首先将文件通过DMA(direct memory access,直接内存拷贝,不经过CPU) 拷贝到系统内核的buffer中,然后在内核buffer中通过 CPU拷贝到用户的buffer中,这才完成了读取文件的步骤。即前四行就发生了两次拷贝。

  • 写数据的时候,将数据从用户buffer通过CPU拷贝到socket buffer中,最后从socket buffer通过DMA拷贝到协议栈。即最后一行也发生了两次拷贝。

整个过程,发生了四次拷贝,三次状态的切换。从一开始的用户态,切换到内核态,再切换到用户态,最后再切换成内核态。一次简单的读写,就有这么多名堂,性能肯定是不好的,所有就出现了零拷贝,零拷贝,不是不拷贝,而是整个过程不需要进行CPU拷贝。


欢迎大家关注我的公众号 javawebkf,目前正在慢慢地将简书文章搬到公众号,以后简书和公众号文章将同步更新,且简书上的付费文章在公众号上将免费。


二、零拷贝

1、使用mmap优化上述流程:
mmap,是指通过内存映射,将文件映射到内核缓冲区,同时,用户空间可以共享内核空间的数据,这样,在进行网络传输时,就可以减少内核空间到用户空间的拷贝次数。同样做上面的事情,使用mmap时整个过程如下:

  • 首先通过DMA拷贝将硬盘数据拷贝到内核buffer,但是因为用户buffer可以共享内核buffer的数据,所以步骤二的cpu拷贝就免了;

  • 然后是直接从内核buffer通过CPU拷贝到socket buffer,最后DMA拷贝到协议栈。

整个过程三次拷贝,三次状态的切换,相比传统拷贝,优化了一丢丢,但这并不是零拷贝。

2、使用sendFile优化:
linux 2.1的sendFile:
sendFile是linux2.1版本开始提供的一个函数,可以让文件直接从内核buffer进入到socket buffer,不需要经过用户态,过程如下:

  • 首先还是将数据从硬盘中通过DMA拷贝到内核buffer,然后通过CPU拷贝将数据从内核buffer拷贝到socket buffer,最后通过DMA拷贝到协议栈。

整个过程还是3次拷贝,但是减少了一次装态切换,从用户态到内核态再到用户态,只经过了两次切换。这里还是有一次CPU拷贝,还不是真正的零拷贝。

linux 2.4的sendFile:
linux 2.4对sendFile又做了一些优化,首先还是DMA拷贝到内核buffer,然后再通过CPU拷贝到socket buffer,最后DMA拷贝到协议栈。优化的点就在于,这次的CPU拷贝,拷贝的内容很少,只拷贝内核buffer的长度、偏移量等信息,消耗很低,可以忽略。因此,这个就是零拷贝。NIO的transferTo方法就可以实现零拷贝。

三、案例代码

1、传统IO拷贝大文件:

  • 服务端:接收客户端发来的数据
public class OldIoServer {
    
    @SuppressWarnings("resource")
    public static void main(String[] args) throws IOException {
        ServerSocket serverSocket = new ServerSocket(6666);
        while (true) {
            Socket socket = serverSocket.accept();
            DataInputStream dataInputStream = new DataInputStream(socket.getInputStream());
            byte[] byteArray = new byte[4096];
            while (true) {
                int readCount = dataInputStream.read(byteArray, 0, byteArray.length);
                if (-1 == readCount) {
                    break;
                }
            }
        }
    }
}
  • 客户端:发送一个大文件到服务端:
public class OldIoClient {
    
    @SuppressWarnings("resource")
    public static void main(String[] args) throws Exception {
        Socket socket = new Socket("127.0.0.1", 6666);
        // 需要拷贝的文件
        String fileName = "E:\\download\\soft\\windows\\jdk-8u171-windows-x64.exe";
        InputStream inputStream = new FileInputStream(fileName);
        DataOutputStream dataOutputStream = new DataOutputStream(socket.getOutputStream());
        byte[] buffer = new byte[4096];
        long readCount;
        long total = 0;
        long start = System.currentTimeMillis();
        while ((readCount = inputStream.read(buffer)) >= 0) {
            total += readCount;
            dataOutputStream.write(buffer);
        }
        long end = System.currentTimeMillis();
        System.out.println("传输总字节数:" + total + ",耗时:" + (end - start) + "毫秒");
        dataOutputStream.close();
        inputStream.close();
        socket.close();
    }
}

这里拷贝了一个JDK,最后运行结果如下:

传输总字节数:217342912,耗时:4803毫秒

可以看到,将近5秒钟。接下来看看使用NIO的transferTo方法耗时情况:

  • 服务端:
public class NioServer {

    public static void main(String[] args) throws IOException {
        InetSocketAddress address = new InetSocketAddress(6666);
        ServerSocketChannel serverSocketChannel = ServerSocketChannel.open();
        ServerSocket serverSocket = serverSocketChannel.socket();
        serverSocket.bind(address);
        ByteBuffer buffer = ByteBuffer.allocate(4096);
        while (true) {
            SocketChannel socketChannel = serverSocketChannel.accept();
            int readCount = 0;
            while (-1 != readCount) {
                readCount = socketChannel.read(buffer);
                buffer.rewind(); // 倒带,将position设置为0,mark设置为-1
            }
        }
    }
}
  • 客户端:
public class NioClient {
    
    @SuppressWarnings("resource")
    public static void main(String[] args) throws IOException {
        SocketChannel socketChannel = SocketChannel.open();
        socketChannel.connect(new InetSocketAddress("127.0.0.1", 6666));
        String fileName = "E:\\download\\soft\\windows\\jdk-8u171-windows-x64.exe";
        FileChannel channel = new FileInputStream(fileName).getChannel();
        long start = System.currentTimeMillis();
        // 在linux下,transferTo方法可以一次性发送数据
        // 在windows中,transferTo方法传输的文件超过8M得分段
        long totalSize = channel.size();
        long transferTotal = 0;
        long position = 0;
        long count = 8 * 1024 * 1024;
        if (totalSize > count) {
            BigDecimal totalCount = new BigDecimal(totalSize).divide(new BigDecimal(count)).setScale(0, RoundingMode.UP);
            for (int i=1; i<=totalCount.intValue(); i++) {
                if (i == totalCount.intValue()) {
                    transferTotal += channel.transferTo(position, totalSize, socketChannel);
                } else {
                    transferTotal += channel.transferTo(position, count + position, socketChannel);
                    position = position + count;
                }
            }
        } else {
            transferTotal += channel.transferTo(position, totalSize, socketChannel);
        }
        
        long end = System.currentTimeMillis();
        System.out.println("发送的总字节:" + transferTotal + ",耗时:" + (end - start) + "毫秒");
        channel.close();
        socketChannel.close();
    }
}

客户端发送文件调用transferTo方法要注意,在window中,这个方法一次只能传输8M,超过8M的文件要分段,像代码中那样分段传输,在linux中是没这个限制的。运行后结果如下:

发送的总字节:217342912,耗时:415毫秒

从结果可以看到,BIO与NIO耗时相差一个数量级,NIO只要0.4s,而BIO要4s。所以在网络传输中,使用NIO的零拷贝,可以大大提高性能。

相关文章

  • NIO与零拷贝

    一、是什么 先来看如下一段代码: 这段代码就是读取一个文件,然后再把它写出去,看起来就几行代码,其实涉及到多次拷贝...

  • NIO与零拷贝

    1 零拷贝原理 1.1 传统IO4次拷贝3次切换 2.1优化后3次拷贝2次切换 零拷贝是从操作系统角度看的,是没有...

  • NIO零拷贝

    先来偷个图,然后扶好小板凳,别摔着了。这是传统IO的文件操作 假设有以下场景,浏览器请求磁盘上的文件a.txt,那...

  • NIO & 零拷贝

    DMA:direct memory access,直接内存拷贝,不使用CPU

  • Netty之二NIO与零拷贝

    个人专题目录 1. Nio与零拷贝 零拷贝是服务器网络编程的关键,任何性能优化都离不开。在 Java 程序员的世界...

  • JVM

    直接内存 使用场景:Unsafe类、NIO零拷贝、Netty的零拷贝、JNI 优点:性能更高 缺点:内存泄漏难排查...

  • NIO-零拷贝

    首先,调用read时,数据文件A拷贝到了kernel模式; 之后,CPU控制将kernel模式数据复制到user模...

  • Netty ByteBuf

    与NIO的ByteBuffer相比可以自定义缓冲区。零拷贝。容量按需增长。读与写模式不需要flip()方法。读和写...

  • 什么是 NIO?

    1、前言 想必大家肯定被什么公众号什么非阻塞、零拷贝、NIO 之类的搞的头了吧,但其实 Java 的 NIO 概念...

  • Nio中文件映射和零拷贝

    NIO中文件映射和零拷贝 Zero-Copy DMA从拷贝至内核缓冲区 cpu将数据从内核缓冲区拷贝至内核空间(s...

网友评论

    本文标题:NIO与零拷贝

    本文链接:https://www.haomeiwen.com/subject/uiqfkktx.html