美文网首页
Java IO与NIO技术体系分析

Java IO与NIO技术体系分析

作者: 东升的思考 | 来源:发表于2017-08-01 23:52 被阅读0次

java.io包中最为核心的一个概念是流(Stream),面向流的编程。流分为两种输入流和输出流,Java中的io流不可能既是输入流,又是输出流。同时,IntputStream和OutputStream都是抽象类, Java中是单继承方式,所以也就不可能出现同时是输入和输出的流。

java.nio中拥有3个核心的概念:Selector、Channel与Buffer。在java.nio中,我们是面向块(block)或者缓冲区(buffer)编程的。buffer本身其实就是一块内存,底层实现上实际是个数组,数据的读、写都是通过buffer来实现的。

如下图所示进行理解,有一个Thread线程,这个线程对应一个Selector,Selector下面有三个Channel,每个Channel下对应着一个Buffer。 一个线程可以在三个Channel通道上来回的切换,通过事件来判断某一时刻在哪一个Channel上执行,因此,事件也是个很重要的概念。

image.png

将数据从Channel读到buffer中,数据已进入内存中,程序中切换写模式到读模式后(通过flip方法实现的),从buffer中将数据读取出来。绝对不会出现直接向Channel写入数据的情况,或者直接从Channel读取数据的情况。
Channel指的是可以向其中写入数据或者从中读取数据的对象,我们可以将nio中的Channel近似理解为io中的Stream。

除了数组之外,Buffer还提供了对数据的结构化访问方式,并且可以追踪到数据的读、写过程。
Java中的7种原生数据类型(不包括boolean类型的buffer)都有各自的对应的Buffer类型,比如IntBuffer、LongBuffer、ByteBuffer以及CharBuffer等等。

与Stream不同的是,Channel是双向的,一个流只可能是IntputStream或者OutputStream,Channel打开后则可以进行读、写或者读写。

由于Channel是双向的,因此,它能更好的反映出底层操作系统的真实情况。
比如说在linux系统中,底层操作系统的通道就是双向的。

好了,上面是一些理论,下面是3个实际的例子进一步结合理论加深理解。

1)随机生成10个数字,然后写入IntBuffer中,最终再从IntBuffer读取并输出。

public class NioTest1 {
    public static void main(String[] args) {
        IntBuffer intbuffer = IntBuffer.allocate(10);

        for (int i =0 ;i < intbuffer.capacity(); i++) {
            int randomNum = new java.security.SecureRandom().nextInt(20);
            intbuffer.put(randomNum);
        }

        intbuffer.flip();  // 切换为读模式

        while (intbuffer.hasRemaining()) {
            System.out.println(intbuffer.get());
        }
    }
}

2)从文件读取内容到Buffer中,然后从Buffer中读取数据输出。

public class NioTest2 {

    public static void main(String[] args) throws Exception {
       try( FileInputStream in = new FileInputStream("niotext2.txt");) {
           FileChannel fileChannel = in.getChannel();

           ByteBuffer byteBuffer = ByteBuffer.allocate(512);
           fileChannel.read(byteBuffer); // 写入byteBuffer

           byteBuffer.flip(); // 切换为读模式

           StringBuffer str = new StringBuffer();
           while (byteBuffer.hasRemaining()) {
               byte b = byteBuffer.get();
               str.append((char)b);
           }
           System.out.println(str.toString());
       }
    }
}

3)将字符数据写入ByteBuffer中,在从ByteBuffer读取写入到文件中。

public class NioTest3 {

    public static void main(String[] args) throws Exception {
        try (FileOutputStream fileOutputStream = new FileOutputStream("niotest3.txt");) {
            FileChannel fileChannel = fileOutputStream.getChannel();

            ByteBuffer byteBuffer = ByteBuffer.allocate(512);

            String str = "hello world, welcom!";
            for (int i =0; i< str.length(); i++) {
                char c = str.charAt(i);
                byteBuffer.put((byte) c);
            }

            byteBuffer.flip(); // 切换为写模式

            fileChannel.write(byteBuffer);
        }
    }
}

相关文章

  • Java IO与NIO技术体系分析

    java.io包中最为核心的一个概念是流(Stream),面向流的编程。流分为两种输入流和输出流,Java中的io...

  • 3 java的IO

    java nio Java的IO体系:旧IO新IO:nio,用ByteBuffer和FileChannel读写ni...

  • Java的IO & NIO

    IO流学习总结一Java IO,硬骨头也能变软二java IO体系的学习总结三Java IO面试题 NIO与AIO...

  • Java NIO

    # Java NIO # Java NIO属于非阻塞IO,这是与传统IO最本质的区别。传统IO包括socket和文...

  • note

    Java IO,NIO,NIO2 以及与操作系统,磁盘 IO NIO模型selector NIO的核心是IO线程池...

  • 大小厂必面Java基础题

    问题:Java 提供了哪些 IO 方式? NIO 如何实现多路复用? 分析:在实际面试中,从传统 IO 到 NIO...

  • NIO之十三-BIO VS NIO

    Java NIO vs. IO Main Differences Betwen Java NIO and IO S...

  • 29、 Java IO与 NIO的区别(补充)

    Java IO与 NIO的区别(补充) NIO即New IO,这个库是在JDK1.4中才引入的。NIO和IO有相同...

  • java-nio学习之java io比较

    java io 基本介绍 java nio和io区别 面向流与面向缓冲 IO是面向流的,NIO是面向缓冲区的。 J...

  • Java Nio中的缓冲区与通道

    Java NIO简介 Java NIO(New IO)是用于Java(来自Java 1.4)的替代IO API,意...

网友评论

      本文标题:Java IO与NIO技术体系分析

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