• 文件IO流总结


      文件在网络上或不同设备之间是怎么传输的,在Java程序中又是怎么来实现文件的传输,带着这两个问题,来了解一下Java中的IO流相关类及操作。

    一、什么是流及流的用途

      流是一组有顺序,有起点和终点的字节的集合,是对数据传输的总称和抽象。简单说流就是在不同设备之间进行数据传输。流的本质是数据传输,JDK为了方便开发者操作流,根据数据传输的各种特性,将流抽象为多种类,从而更加方便直观的操作。

    二、流的分类

      根据处理的数据类型的不同,可将IO流分为字节流和字符流;根据IO流的流向又可将其分为输入流和输出流。一般来说,如果没有指出按什么分类,IO流的分类默认按处理数据的类型分为:字符输入流、字符输出流、字节输入流及字节输出流

    三、字符流和字节流的区别

      字符流的由来: 因为数据编码不同,而有了对字符进行高效操作的流对象。字符流本质是基于字节流读取,并查询指定的码表。

    l  读写单位不同:字节流以字节(8bit)为单位,字符流以字符为单位,根据码表映射字符,一次可能读多个字节。

    l  处理对象不同:字节流能处理所有类型的数据(如图片、avi等),而字符流只能处理字符类型的数据。

    结论:只要是处理纯文本数据,优先考虑使用字符流。 除此之外都使用字节流。

    四、 输入流和输出流

      输入和输出相对于内存而言,输入读入内存,输出从内存输出。对输入流只能进行读操作,对输出流只能进行写操作。

    五、 流对象

     1) 字符输入流   Reader

    1. Reader 是所有的输入字符流的父类,它是一个抽象类;
    2. CharReader、StringReader 是两种基本的介质流,它们分别将Char 数组、String中读取数据。PipedReader 是从与其它线程共用的管道中读取数据;
    3. BufferedReader 很明显就是一个装饰器,它和其子类负责装饰其它Reader 对象;
    4. FilterReader 是所有自定义具体装饰流的父类,其子类PushbackReader 对Reader 对象进行装饰,会增加一个行号;
    5. InputStreamReader 是一个连接字节流和字符流的桥梁,它将字节流转变为字符流。FileReader 可以说是一个达到此功能、常用的工具类,在其源代码中明显使用了将FileInputStream 转变为Reader 的方法。我们可以从这个类中得到一定的技巧。Reader 中各个类的用途和使用方法基本和InputStream 中的类使用一致。后面会有Reader 与InputStream 的对应关系。

     2)字符输出流   Writer

    1. Writer 是所有的输出字符流的父类,它是一个抽象类;
    2. CharArrayWriter、StringWriter 是两种基本的介质流,它们分别向Char 数组、String 中写入数据。PipedWriter 是向与其它线程共用的管道中写入数据;
    3. BufferedWriter 是一个装饰器为Writer 提供缓冲功能;
    4. PrintWriter 和PrintStream 极其类似,功能和使用也非常相似;
    5. OutputStreamWriter 是OutputStream 到Writer 转换的桥梁,它的子类FileWriter 其实就是一个实现此功能的具体类(具体可以研究一SourceCode)。功能和使用和OutputStream 极其类似,后面会有它们的对应图。

     3)字节输入流   InputStream

    1. InputStream 是所有的输入字节流的父类,它是一个抽象类;
    2. ByteArrayInputStream、StringBufferInputStream、FileInputStream 是三种基本的介质流,它们分别从Byte 数组、StringBuffer、和本地文件中读取数据。PipedInputStream 是从与其它线程共用的管道中读取数据,与Piped 相关的知识后续单独介绍;
    3. ObjectInputStream 和所有FilterInputStream 的子类都是装饰流(装饰器模式的主角)。

     4)字节输出流   OutputStream

    1. OutputStream 是所有的输出字节流的父类,它是一个抽象类;
    2. ByteArrayOutputStream、FileOutputStream 是两种基本的介质流,它们分别向Byte 数组、和本地文件中写入数据。PipedOutputStream 是向与其它线程共用的管道中写入数据;
    3. ObjectOutputStream 和所有FilterOutputStream 的子类都是装饰流。

    六、转换流

     


       1)具体实现对象类

    1. InputStreamReader        字节到字符的桥梁
    2. OutputStreamWriter       字符到字节的桥梁

    注意:这两个流对象是字符体系中的成员,本身是字符流,所以在构造的时候需要传入字节流对象。

     2)特点:

    1. 字符流和字节流之间的桥梁
    2. 可对读取到的字节数据经过指定编码转换成字符
    3. 可对读取到的字符数据经过指定编码转换成字节

     3)什么时候使用转换流

    1. 当字节和字符之间有转换动作时;
    2. 流操作的数据需要编码或解码时。

    七、File

      File类是对文件系统中文件以及文件夹进行封装的对象,可以通过面向对象的思想来操作文件和文件夹。 File类保存文件或目录的各种元数据信息,包括文件名、文件长度、最后修改时间、是否可读、获取当前文件的路径名,判断指定文件是否存在、获得当前目录中的文件列表,创建、删除文件和目录等方法。

    八、RandomAccessFile

      该对象并不是流体系中的一员,其封装了字节流,同时还封装了一个缓冲区(字符数组),通过内部的指针来操作字符数组中的数据。 该对象特点:
     1)该对象只能操作文件,所以构造函数接收两种类型的参数:a.字符串文件路径;b.File对象。
     2)该对象既可以对文件进行读操作,也能进行写操作,在进行对象实例化时可指定操作模式(r,rw)
      注意:该对象在实例化时,如果要操作的文件不存在,会自动创建;如果文件存在,写数据未指定位置,会从头开始写,即覆盖原有的内容。 可以用于多线程下载或多个线程同时写数据到文件。
      看完了上面的总结,不知道亲是否对IO流的操作有个整体的印象,接下来还是通过代码来体现一下面向对象的思想吧。

    package cn.dolphin.io;
    
    import java.io.BufferedInputStream;
    import java.io.BufferedOutputStream;
    import java.io.BufferedReader;
    import java.io.BufferedWriter;
    import java.io.File;
    import java.io.FileInputStream;
    import java.io.FileOutputStream;
    import java.io.FileReader;
    import java.io.FileWriter;
    import java.io.IOException;
    import java.io.InputStreamReader;
    import java.io.OutputStreamWriter;
    import java.io.PrintStream;
    //import java.io.PrintWriter;
    
    /**
     * Java中IO操作的演示
     * 
     * @author Cyanide
     * @version v1.0.0
     * @since v1.0.0
     * 
     */
    public class FileDemo {
        public static void main(String[] args) throws IOException {
            // createFile();
            // copyText1();
            // copyText2();
            // copyText3();
            // copyText4();
            // copyText5();
            // copyBinary1();
            // copyBinary2();
            // copyBinary3();
            //copyBinary4();
            //codec();
            printFlow();
        }
    
        // -------------------------------------------------//
        //             基本类输入输出流对象及高效缓冲流对象            //
        // -------------------------------------------------//
        /**
         * File类的文件目录创建及删除操作演示
         * 
         * @throws IOException
         */
        static void createFile() throws IOException {
            File f1 = new File("file.txt");
            File f2 = new File("directory");
            // 在当前项目目录中创建文件,需要处理异常。
            f1.createNewFile();
            // 在当前项目目录中创建目录
            f2.mkdir();
            // 删除创建的目录的文件
            f1.delete();
            f2.delete();
        }
    
        /**
         * 文本文件的复制操作,基本实现,每读一个字符写一次。
         * 
         * @throws IOException
         */
        static void copyText1() throws IOException {
            // 创建FileReader对象,抛出FileNotFoundException。
            FileReader fileReader = new FileReader("file.txt");
            // 创建FileWriter对象,抛出IOException。
            FileWriter fileWriter = new FileWriter("copies.txt");
            int num = 0;// fileReader.read()读到文件末尾返回-1。
            while ((num = fileReader.read()) != -1) {
                fileWriter.write(num);
            }
            fileWriter.close();
            fileReader.close();
            // 点评:效率低下,因此会采用第二种方式进行拷皮。
        }
    
        /**
         * 文本文件的复制操作,每读1024个字符写一次。
         * 
         * @throws IOException
         */
        static void copyText2() throws IOException {
            FileReader fileReader = new FileReader("file.txt");
            FileWriter fileWriter = new FileWriter("copies.txt");
            // length每次读出字符的实际长度
            int length = 0;
            char[] chs = new char[1024];
            while ((length = fileReader.read(chs)) != -1) {
                // 注意:fileReader.read()将读到的字符存入数组,如果最后
                // 一次读到字符长度不足字符数组长度时,未覆盖索引处字符也会
                // 读出,因此指定 length,读多少,写多少。
                fileWriter.write(chs, 0, length);
            }
            fileWriter.close();
            fileWriter.close();
            // 点评:效率比copyText1()高出很多,但还有更好的。
        }
    
        /**
         * 文本文件的复制操作,高效缓存读写,每次一个字符。
         * 
         * @throws IOException
         */
        static void copyText3() throws IOException {
            BufferedReader bufferedReader = new BufferedReader(new FileReader(
                    "file.txt"));
            BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(
                    "copies.txt"));
            int num = 0;
            while ((num = bufferedReader.read()) != -1) {
                bufferedWriter.write(num);
            }
            bufferedWriter.close();
            bufferedReader.close();
        }
    
        /**
         * 文本文件的复制操作,高效缓存读写,每次读1024个字符。
         * 
         * @throws IOException
         */
        static void copyText4() throws IOException {
            BufferedReader bufferedReader = new BufferedReader(new FileReader(
                    "file.txt"));
            BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(
                    "copies.txt"));
            int length = 0;
            char[] chs = new char[1024];
            while ((length = bufferedReader.read(chs)) != -1) {
                bufferedWriter.write(chs, 0, length);
            }
            bufferedWriter.close();
            bufferedReader.close();
        }
    
        /**
         * 文本文件的复制操作,高效缓存读写,每次读一行,特有方法读写。
         * 
         * @throws IOException
         */
        static void copyText5() throws IOException {
            BufferedReader bufferedReader = new BufferedReader(new FileReader(
                    "file.txt"));
            BufferedWriter bufferedWriter = new BufferedWriter(new FileWriter(
                    "copies.txt"));
            String str = null;
            while ((str = bufferedReader.readLine()) != null) {
                bufferedWriter.write(str);
                // readLine()方法读取不带换行,所以需要写入换行。
                bufferedWriter.newLine();
                bufferedWriter.flush();
            }
            bufferedWriter.close();
            bufferedReader.close();
        }
    
        /**
         * 二进制文件的复制操作,基本实现,每读一个字节写一次。
         * 
         * @throws IOException
         */
        static void copyBinary1() throws IOException {
            FileInputStream fileInputStream = new FileInputStream("file.bmp");
            FileOutputStream fileOutputStream = new FileOutputStream("copies.bmp");
            int num = 0;
            while ((num = fileInputStream.read()) != -1) {
                fileOutputStream.write(num);
            }
            fileInputStream.close();
            fileOutputStream.close();
        }
    
        /**
         * 二进制文件的复制操作,每读1024个字节写一次。
         * 
         * @throws IOException
         */
        static void copyBinary2() throws IOException {
            FileInputStream fileInputStream = new FileInputStream("file.bmp");
            FileOutputStream fileOutputStream = new FileOutputStream("copies.bmp");
            int length = 0;
            byte[] bytes = new byte[1024];
            while ((length = fileInputStream.read(bytes)) != -1) {
                fileOutputStream.write(bytes, 0, length);
            }
            fileInputStream.close();
            fileOutputStream.close();
        }
    
        /**
         * 二进制文件的复制操作,高效缓存读写,每次一个字节。
         * 
         * @throws IOException
         */
        static void copyBinary3() throws IOException {
            BufferedInputStream bufferedInputStream = new BufferedInputStream(
                    new FileInputStream("file.bmp"));
            BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(
                    new FileOutputStream("copies.bmp"));
            int num = 0;
            while ((num = bufferedInputStream.read()) != -1) {
                bufferedOutputStream.write(num);
            }
            bufferedOutputStream.close();
            bufferedInputStream.close();
        }
    
        /**
         * 二进制文件的复制操作,高效缓存读写,每次读1024个字节。
         * 
         * @throws IOException
         */
        static void copyBinary4() throws IOException {
            BufferedInputStream bufferedInputStream = new BufferedInputStream(
                    new FileInputStream("file.bmp"));
            BufferedOutputStream bufferedOutputStream = new BufferedOutputStream(
                    new FileOutputStream("copies.bmp"));
            int length = 0;
            byte[] bytes = new byte[1024];
            while ((length = bufferedInputStream.read(bytes)) != -1) {
                bufferedOutputStream.write(bytes, 0, length);
            }
            bufferedInputStream.close();
            bufferedOutputStream.close();
        }
    
        // -------------------------------------------------//
        //                 缓冲高效转换流对象                        //
        // -------------------------------------------------//
        /**
         * 高效缓冲流结合转换流对文本进行复制编码及解码过程。
         * @throws IOException
         */
        static void codec() throws IOException {
            //编码过程,将文本文件中编码。
            BufferedReader bufferedReader = new BufferedReader(
                    new InputStreamReader(new FileInputStream("file.txt"), "gb2312"));
            //解码过程,将解码后的流写入文本文件。
            //BufferedWriter bufferedWriter = new BufferedWriter(
                    //new OutputStreamWriter(new FileOutputStream("copies.txt"),"gb2312"));
            //将键盘录入内容编码
            //System.out.print("输入测试文本内容:");
            //BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in,"gb2312"));
            //将解码后的流直接输出到Console。
            BufferedWriter bufferedWriter = new BufferedWriter(new OutputStreamWriter(System.out,"utf-8"));
            int length = 0;
            char[] chs = new char[1024];
            while ((length = bufferedReader.read(chs))!=-1) {
                bufferedWriter.write(chs,0,length);
            }
            bufferedWriter.close();
            bufferedReader.close();
            //总结:用什么编码,就用什么解码。 如果编码和解码不一致,如果上例就会出现乱码显示。
        }
        
        static void printFlow() throws IOException{
            BufferedReader bufferedReader = new BufferedReader(new FileReader("file.txt"));
            //将流写入文本。
            //PrintWriter printWriter = new PrintWriter(new FileWriter("copies.txt"),true);
            //将流输出到Console。
            PrintStream printStream = new PrintStream(System.out);
            char[] chs = new char[1024];
            while (bufferedReader.read(chs)!=-1) {
                printStream.println(chs);
                //printWriter.println(chs);
                bufferedReader.read(chs);
            }
            bufferedReader.close();
            printStream.close();
        }
    }

      最后PrintStream及PrintWriter只是输出,没有相应的输入。想想Java开始的第一天那个HelloWorld就用到了PrintStream只是刚开始不明白罢了。或许现在了还有不明白的地方,但随着知识的积累,相信学习Java会是件很快乐的事。

  • 相关阅读:
    Android开源日志框架xlog
    [CrackMe]160个CrackMe之18
    SEH异常
    全局句柄表
    用户层异常的派发与处理
    用户层异常的处理
    内核层异常的收集与处理
    两种异常(CPU异常、用户模拟异常)的收集
    无处不在的页异常
    AppBoxFuture(四). 随需而变-Online Schema Change
  • 原文地址:https://www.cnblogs.com/magics/p/3648791.html
Copyright © 2020-2023  润新知