java io

设计模式

装饰者模式、适配器模式java

类别

同步阻塞IO(BIO)

BIO是一个链接一个线程。面试

传统的同步阻塞模型开发中,ServerSocket负责绑定IP地址,启动监听端口;Socket负责发起链接操做。链接成功后,双方经过输入和输出流进行同步阻塞式通讯。编程

该模型最大的问题就是缺少弹性伸缩能力,当客户端并发访问量增长后,服务端的线程个数和客户端并发访问数呈1:1的正比关系,Java中的线程也是比较宝贵的系统资源,线程数量快速膨胀后,系统的性能将急剧降低,随着访问量的继续增大,系统最终就死-掉-了设计模式

同步非阻塞IO(NIO)

NIO提供了与传统BIO模型中的Socket和ServerSocket相对应的SocketChannel和ServerSocketChannel两种不一样的套接字通道实现。两种通道都支持阻塞和非阻塞两种模式。数组

对于低负载、低并发的应用程序,可使用同步阻塞I/O来提高开发速率和更好的维护性;对于高负载、高并发的(网络)应用,应使用NIO的非阻塞模式来开发。安全

  • 缓冲区服务器

    java.nio.Buffer是一个抽象类。网络

    在NIO库中,全部数据都是用缓冲区处理的,缓冲区包含一些要写入或者读出的数据。数据结构

    缓冲区其实是一个数组,并提供了对数据结构化访问以及维护读写位置等信息。如:ByteBuffer、CharBuffer、 ShortBuffer、IntBuffer、LongBuffer、FloatBuffer、DoubleBuffer。并发

  • 通道

    咱们对数据的读取和写入要经过Channel,它就像水管同样,是一个通道。通道不一样于流的地方就是通道是双向的,能够用于读、写和同时读写操做。

    底层的操做系统的通道通常都是全双工的,因此全双工的Channel比流能更好的映射底层操做系统的API。

    • ​ SelectableChannel:用户网络读写
    • ​ FileChannel:用于文件操做
  • 多路复用器

    Selector提供选择已经就绪的任务的能力:Selector会不断轮询注册在其上的Channel,若是某个Channel上面发生读或者写事件,这个Channel就处于就绪状态,会被Selector轮询出来,而后经过SelectionKey能够获取就绪Channel的集合,进行后续的I/O操做。

    一个Selector能够同时轮询多个Channel,由于JDK使用了epoll()代替传统的select实现,因此没有最大链接句柄1024/2048的限制。因此,只须要一个线程负责Selector的轮询,就能够接入成千上万的客户端。

    • select和poll具备O(n)的无差异轮询复杂度
    • epoll事件复杂度为O(1),基于时间驱动的

NIO是一个请求一个线程,即客户端发送的链接请求都会注册到多路复用器上,多路复用器轮询到链接有I/O请求时才启动一个线程进行处理。

适用场景:聊天服务器

异步非阻塞IO(AIO)

NIO 2.0引入了新的异步通道的概念,并提供了异步文件通道和异步套接字通道的实现。

异步的套接字通道是真正的异步非阻塞I/O,对应于UNIX网络编程中的事件驱动I/O(AIO)。他不须要过多的Selector对注册的通道进行轮询便可实现异步读写,从而简化了NIO的编程模型。

  • AsynchronousSocketChannel
  • AsynchronousServerSocketChannel
  • AsynchronousFileChannel
  • AsynchronousDatagramChannel

AIO是一个有效请求一个线程,客户端的I/O请求都是由OS先完成了再通知服务器应用去启动线程进行处理。

适用场景:相册服务器

根据处理数据类型分类

流式

主体部分

字节流

按字节读,可用于文件、图片、视频、音频

OutputStream、InputStream

字符流

按字符读,通常用于文件

Writer、Reader

Java中字符是采用Unicode标准,一个字符是16位,即一个字符使用两个字节来表示。为此,JAVA中引入了处理字符的流。由于数据编码的不一样,而有了对字符进行高效操做的流对象。本质其实就是基于字节流读取时,去查了指定的码表。

字节流没有缓冲区,是直接输出的,而字符流是输出到缓冲区的。所以在输出时,字节流不调用close()方法时,信息已经输出了,而字符流只有在调用close()方法关闭缓冲区时,信息才输出。要想字符流在未关闭时输出信息,则须要手动调用flush()方法。

非流式

辅助流式部分的类,如:File、RandomAccessFile、FileDescriptor

其余类

文件读取部分的与安全相关的类,如:SerializablePermission;

与本地操做系统相关的文件系统的类,如:FileSystem、Win32FileSystem、WinNTFileSystem。

根据数据来源/操做对象分类

  • 一、文件(file):FileInputStream、FileOutputStream、FileReader、FileWriter
  • 二、数组([]):
    • 2.一、字节数组(byte[]):ByteArrayInputStream、ByteArrayOutputStream
    • 2.二、字符数组(char[]):CharArrayReader、CharArrayWriter
  • 三、管道操做:PipedInputStream、PipedOutputStream、PipedReader、PipedWriter
  • 四、基本数据类型:DataInputStream、DataOutputStream
  • 五、缓冲操做:BufferedInputStream、BufferedOutputStream、BufferedReader、BufferedWriter
  • 六、打印:PrintStream、PrintWriter
  • 七、对象序列化反序列化:ObjectInputStream、ObjectOutputStream
  • 八、转换:InputStreamReader、OutputStreWriter
  • 九、字符串(String)Java8中已废弃StringBufferInputStream、StringBufferOutputStream、StringReader、StringWriter

面试题

  1. 什么是IO流?
    它是一种数据的流从源头流到目的地。好比文件拷贝,输入流和输出流都包括了。输入流从文件中读取数据存储到进程(process)中,输出流从进程中读取数据而后写入到目标文件。

  2. 字节流和字符流的区别。
    字节流在JDK1.0中就被引进了,用于操做包含ASCII字符的文件。JAVA也支持其余的字符如Unicode,为了读取包含Unicode字符的文件,JAVA语言设计者在JDK1.1中引入了字符流。ASCII做为Unicode的子集,对于英语字符的文件,能够可使用字节流也可使用字符流。

  3. Java中流类的超类主要由那些?

    • java.io.InputStream
    • java.io.OutputStream
    • java.io.Reader
    • java.io.Writer
  4. FileInputStream和FileOutputStream是什么?
    这是在拷贝文件操做的时候,常常用到的两个类。在处理小文件的时候,它们性能表现还不错,在大文件的时候,最好使用BufferedInputStream (或 BufferedReader) 和 BufferedOutputStream (或 BufferedWriter)

    public class InputAndOutputBuffering {
        public static void main(String args[]) throws IOException {
            FileInputStream fistream = new FileInputStream("pqr.txt");
            BufferedInputStream bistream = new BufferedInputStream(fistream);
    
            FileOutputStream fostream = new FileOutputStream("xyz.txt");
            BufferedOutputStream bostream = new BufferedOutputStream(fostream);
    
            int temp;
            while( ( temp = bistream.read() ) != -1 ) {
                bostream.write(temp);
                System.out.print((char) temp);
            }
            bostream.close();
            fostream.close();
            bistream.close();
            fistream.close();
        }
    }
  5. 字节流和字符流,你更喜欢使用拿一个?
    我的来讲,更喜欢使用字符流,由于他们更新一些。许多在字符流中存在的特性,字节流中不存在。好比使用BufferedReader而不是BufferedInputStreams或DataInputStream,使用newLine()方法来读取下一行,可是在字节流中咱们须要作额外的操做。

  6. System.out.println()是什么?
    println是PrintStream的一个方法。out是一个静态PrintStream类型的成员变量,System是一个java.lang包中的类,用于和底层的操做系统进行交互。

  7. 什么是Filter流?
    Filter Stream是一种IO流主要做用是用来对存在的流增长一些额外的功能,像给目标文件增长源文件中不存在的行数,或者增长拷贝的性能。

  8. 有哪些可用的Filter流?
    在java.io包中主要由4个可用的filter Stream。两个字节filter stream,两个字符filter stream,分别是FilterInputStream, FilterOutputStream, FilterReader and FilterWriter。这些类是抽象类,不能被实例化的。

  9. SequenceInputStream的做用?
    在拷贝多个文件到一个目标文件的时候是很是有用的。可用使用不多的代码实现

    public class TwoFiles {
        public static void main(String args[]) throws IOException {
            FileInputStream fistream1 = new FileInputStream("A.txt");  // first source file
            FileInputStream fistream2 = new FileInputStream("B.txt");  //second source file
    
            SequenceInputStream sistream = new SequenceInputStream(fistream1, fistream2);
            FileOutputStream fostream = new FileOutputStream("C.txt");// destination file
    
            int temp;
            while( ( temp = sistream.read() ) != -1) {
                System.out.print( (char) temp ); // to print at DOS prompt
                fostream.write(temp);   // to write to file
            }
            fostream.close();
            sistream.close();
            fistream1.close();
            fistream2.close();
        }
    }
  10. 说说PrintStream和PrintWriter
    他们两个的功能相同,可是属于不一样的分类。字节流和字符流。他们都有println()方法。

  11. 在文件拷贝的时候,那一种流可用提高更多的性能?
    在字节流的时候,使用BufferedInputStream和BufferedOutputStream。
    在字符流的时候,使用BufferedReader 和 BufferedWriter

  12. 说说管道流(Piped Stream)
    有四种管道流, PipedInputStream, PipedOutputStream, PipedReader 和 PipedWriter.在多个线程或进程中传递数据的时候管道流很是有用。

  13. 说说File类
    它不属于 IO流,也不是用于文件操做的,它主要用于知道一个文件的属性,读写权限,大小等信息。

  14. 说说RandomAccessFile?
    它在java.io包中是一个特殊的类,既不是输入流也不是输出流,它二者均可以作到。他是Object的直接子类。一般来讲,一个流只有一个功能,要么读,要么写。可是RandomAccessFile既能够读文件,也能够写文件。 DataInputStream 和 DataOutStream有的方法,在RandomAccessFile中都存在。

做者:艾贺521 连接:https://www.imooc.com/article/24305

相关文章
相关标签/搜索