目录

IO使用

原文:https://www.cnblogs.com/hopeyes/p/9736642.html

近期学习了Java的IO流,尝试着总结一下。

java.io 包下的IO流很多:

https://gitee.com/lienhui68/picStore/raw/master/null/20200827032501.png

其中,以Stream结尾的为字节流,以Writer或者Reader结尾的为字符流。所有的输入流都是抽象类IuputStream(字节输入流)或者抽象类Reader(字符输入流)的子类,所有的输出流都是抽象类OutputStream(字节输出流)或者抽象类Writer(字符输出流)的子类。字符流能实现的功能字节流都能实现,反之不一定。如:图片,视频等二进制文件,只能使用字节流读写。

字符流FileReader和FileWriter

FileReader类构造方法摘要

1
public class FileReader extends InputStreamReader {
  1. FileReader(File file)

    在给定从中读取数据的 File 的情况下创建一个新 FileReader。

  2. FileReader(FileDescriptor fd)

    在给定从中读取数据的 FileDescriptor 的情况下创建一个新 FileReader。

  3. FileReader(String fileName)

    在给定从中读取数据的文件名的情况下创建一个新 FileReader。

FileWriter类构造方法摘要

1
public class FileWriter extends OutputStreamWriter {
  1. FileWriter(File file)

    根据给定的 File 对象构造一个 FileWriter 对象。

  2. FileWriter(File file, boolean append)

    根据给定的 File 对象构造一个 FileWriter 对象。

  3. FileWriter(FileDescriptor fd)

    构造与某个文件描述符相关联的 FileWriter 对象。

  4. FileWriter(String fileName)

    根据给定的文件名构造一个 FileWriter 对象。

  5. FileWriter(String fileName, boolean append)

    根据给定的文件名以及指示是否附加写入数据的 boolean 值来构造 FileWriter 对象。

使用FileReader和FileWriter类完成文本文件复制

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package com.eh.eden.pattern;

import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;

class CopyFile {
    public static void main(String[] args) throws IOException {
        try (
                //创建输入流对象
                //文件不存在会抛出java.io.FileNotFoundException
                FileReader fr = new FileReader("/tmp/com.eh/copyfrom.txt");
                //创建输出流对象
                FileWriter fw = new FileWriter("/tmp/com.eh/copyto.txt")
                /*
                创建输出流做的工作:
                 1、调用系统资源创建了一个文件
                 2、创建输出流对象
                 3、把输出流对象指向文件
                 */
        ) {
            //文本文件复制,一次读一个字符
//            method1(fr, fw);
            //文本文件复制,一次读一个字符数组
            method2(fr, fw);
        }
    }

    //文本文件复制,一次读一个字符
    public static void method1(FileReader fr, FileWriter fw) throws IOException {
        int ch;
        while ((ch = fr.read()) != -1) { // 读数据
            fw.write(ch); // 写数据
        }
        fw.flush();
    }

    //文本文件复制,一次读一个字符数组
    public static void method2(FileReader fr, FileWriter fw) throws IOException {
        char[] chs = new char[1024];
        int len;
        while ((len = fr.read(chs)) != -1) {
            fw.write(chs, 0, len);
        }
        fw.flush();
    }
}

字符缓冲流BufferedReader和BufferedWriter

字符缓冲流具备文本特有的表现形式,行操作

1
public class BufferedReader extends Reader
  1. 从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。

  2. 可以指定缓冲区的大小,或者可使用默认的大小。大多数情况下,默认值就足够大了。

  3. 通常,Reader 所作的每个读取请求都会导致对底层字符或字节流进行相应的读取请求。因此,建议用 BufferedReader 包装所有其 read() 操作可能开销很高的 Reader(如 FileReader 和 InputStreamReader)。例如,

    1
    
    BufferedReader in = new BufferedReader(new FileReader("foo.in"));
    
  4. 将缓冲指定文件的输入。如果没有缓冲,则每次调用 read() 或 readLine() 都会导致从文件中读取字节,并将其转换为字符后返回,而这是极其低效的。

1
public class BufferedWriter extends Writer
  1. 将文本写入字符输出流,缓冲各个字符,从而提供单个字符、数组和字符串的高效写入。

  2. 可以指定缓冲区的大小,或者接受默认的大小。在大多数情况下,默认值就足够大了。

  3. 该类提供了 newLine() 方法,它使用平台自己的行分隔符概念,此概念由系统属性 line.separator 定义。并非所有平台都使用新行符 ('\n') 来终止各行。因此调用此方法来终止每个输出行要优于直接写入新行符

  4. 通常 Writer 将其输出立即发送到底层字符或字节流。除非要求提示输出,否则建议用 BufferedWriter 包装所有其 write() 操作可能开销很高的 Writer(如 FileWriters 和 OutputStreamWriters)。例如,

    1
    
    PrintWriter out = new PrintWriter(new BufferedWriter(new FileWriter("foo.out")));
    
  5. 缓冲 PrintWriter 对文件的输出。如果没有缓冲,则每次调用 print() 方法会导致将字符转换为字节,然后立即写入到文件,而这是极其低效的。

使用BufferedReader和BufferedWriter完成文件复制

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
package com.eh.eden.pattern;

import java.io.*;

class CopyFile {
    public static void main(String[] args) throws IOException {
        try (
                //创建输入流对象
                //文件不存在会抛出java.io.FileNotFoundException
                BufferedReader br = new BufferedReader(new FileReader("/tmp/com.eh/copyfrom.txt"));
                //创建输出流对象
                BufferedWriter bw = new BufferedWriter(new FileWriter("/tmp/com.eh/copyto.txt"))
        ) {
            //文本文件复制
            char[] chs = new char[1024];
            int len;
            while ((len = br.read(chs)) != -1) {
                bw.write(chs, 0, len);
            }
        }
    }
}

缓冲区的工作原理:

  1. 使用了底层流对象从具体设备上获取数据,并将数据存储到缓冲区的数组内。
  2. 通过缓冲区的read()方法从缓冲区获取具体的字符数据,这样就提高了效率。
  3. 如果用read方法读取字符数据,并存储到另一个容器中,直到读取到了换行符时,将另一个容器临时存储的数据转成字符串返回,就形成了readLine()功能。

字节流FileInputStream和FileOutputStream

FileInputStream

1
public class FileInputStream extends InputStream
  1. FileInputStream 从文件系统中的某个文件中获得输入字节。哪些文件可用取决于主机环境。
  2. FileInputStream 用于读取诸如图像数据之类的原始字节流。

构造方法摘要

  1. FileInputStream(File file)

    通过打开一个到实际文件的连接来创建一个 FileInputStream,该文件通过文件系统中的 File 对象 file 指定。

  2. FileInputStream(FileDescriptor fdObj)

    通过使用文件描述符 fdObj 创建一个 FileInputStream,该文件描述符表示到文件系统中某个实际文件的现有连

  3. FileInputStream(String name)

    通过打开一个到实际文件的连接来创建一个 FileInputStream,该文件通过文件系统中的路径名 name 指定。

FileOutputStream

1
public class FileOutputStream extends OutputStream
  1. 文件输出流是用于将数据写入 File 或 FileDescriptor 的输出流。文件是否可用或能否可以被创建取决于基础平台。特别是某些平台一次只允许一个 FileOutputStream(或其他文件写入对象)打开文件进行写入。在这种情况下,如果所涉及的文件已经打开,则此类中的构造方法将失败。

  2. FileOutputStream 用于写入诸如图像数据之类的原始字节的流。

  3. 构造方法摘要 FileOutputStream(File file)

    创建一个向指定 File 对象表示的文件中写入数据的文件输出流。

  4. FileOutputStream(File file, boolean append)

    创建一个向指定 File 对象表示的文件中写入数据的文件输出流。

  5. FileOutputStream(FileDescriptor fdObj)

    创建一个向指定文件描述符处写入数据的输出文件流,该文件描述符表示一个到文件系统中的某个实际文件的现有连接。

  6. FileOutputStream(String name)

    创建一个向具有指定名称的文件中写入数据的输出文件流。

  7. FileOutputStream(String name, boolean append)

    创建一个向具有指定 name 的文件中写入数据的输出文件流。

使用字节流复制图片

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
package com.eh.eden.pattern;

import java.io.*;

class CopyFile {
    public static void main(String[] args) throws IOException {
        try (
                FileInputStream fin = new FileInputStream("/tmp/com.eh/copyfrom.gif");
                FileOutputStream fout = new FileOutputStream("/tmp/com.eh/copyto.gif")
        ) {
            int len;
            byte[] buff = new byte[1024];
            while ((len = fin.read(buff)) != -1) {
                fout.write(buff, 0, len);
            }
        }
    }
}

字节缓冲流BufferedInputStream和BufferedOutputStream

BufferedInputStream

1
public class BufferedInputStream extends FilterInputStream

BufferedInputStream 为另一个输入流添加一些功能,即缓冲输入以及支持 mark 和 reset 方法的能力。在创建 BufferedInputStream 时,会创建一个内部缓冲区数组。在读取或跳过流中的字节时,可根据需要从包含的输入流再次填充该内部缓冲区,一次填充多个字节。mark 操作记录输入流中的某个点,reset 操作使得在从包含的输入流中获取新字节之前,再次读取自最后一次 mark 操作后读取的所有字节。

BufferedOutputStream

1
public class BufferedOutputStream extends FilterOutputStream

该类实现缓冲的输出流。通过设置这种输出流,应用程序就可以将各个字节写入底层输出流中,而不必针对每次字节写入调用底层系统。

使用字节缓冲流实现图片的复制

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
package com.eh.eden.pattern;

import java.io.*;

class CopyFile {
    public static void main(String[] args) throws IOException {
        try (
                // 默认缓冲区大小8K
                BufferedInputStream bin = new BufferedInputStream(new FileInputStream("/tmp/com.eh/copyfrom.gif"));
                BufferedOutputStream bout = new BufferedOutputStream(new FileOutputStream("/tmp/com.eh/copyto.gif"))
        ) {
            int len;
            byte[] buff = new byte[1024];
            while ((len = bin.read(buff)) != -1) {
                bout.write(buff, 0, len);
            }
        }
    }
}

转换流:InputStreamReader和OutputStreamWriter

InputStreamReader和OutputStreamWriter是字符和字节的桥梁,也可称之为字符转换流。原理:字节流+编码。

FileReader和FileWriter作为子类,仅作为操作字符文件的便捷类存在。当操作的字符文件,使用的是默认编码表时可以不用父类,而直接使用子类完成操作,简化代码。

一旦要指定其他编码时,不能使用子类,必须使用字符转换流。

InputStreamReader

1
public class InputStreamReader extends Reader
  1. InputStreamReader 是字节流通向字符流的桥梁:它使用指定的 charset 读取字节并将其解码为字符。它使用的字符集可以由名称指定或显式给定,或者可以接受平台默认的字符集。

  2. 每次调用 InputStreamReader 中的一个 read() 方法都会导致从底层输入流读取一个或多个字节。要启用从字节到字符的有效转换,可以提前从底层流读取更多的字节,使其超过满足当前读取操作所需的字节。

  3. 为了达到最高效率,可以考虑在 BufferedReader 内包装 InputStreamReader。例如:

    1
    
     BufferedReader in = new BufferedReader(new InputStreamReader(System.in))//重要
    

    使用标准输入流,读取键盘录入的数据,存储到临时目录下的a.txt中

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    
    package com.eh.eden.pattern;
       
    import java.io.*;
       
    /**
     * 使用标准输入流,读取键盘录入的数据,存储到临时目录下的a.txt中
     * 将字节输入流转换成字符输入流,InputStreamReader
     */
    class InputStreamReaderDemo {
        public static void main(String[] args) throws IOException {
            try (
                    // //创建输入流对象
                    BufferedReader br = new BufferedReader(new InputStreamReader(System.in));
                    //创建输出流对象
                    BufferedWriter bw = new BufferedWriter(new FileWriter("/tmp/com.eh/a.txt"))
            ) {
                 // 读写数据
                char[] chs = new char[1024];
                int len;
                while ((len = br.read(chs)) != -1) {
                    bw.write(chs, 0, len);
                    bw.flush();
                }
            }
        }
    }
    

OutputStreamWriter

1
public class OutputStreamWriter extends Writer
  1. OutputStreamWriter 是字符流通向字节流的桥梁:可使用指定的 charset 将要写入流中的字符编码成字节。它使用的字符集可以由名称指定或显式给定,否则将接受平台默认的字符集。

  2. 每次调用 write() 方法都会导致在给定字符(或字符集)上调用编码转换器。在写入底层输出流之前,得到的这些字节将在缓冲区中累积。可以指定此缓冲区的大小,不过,默认的缓冲区对多数用途来说已足够大。注意,传递给 write() 方法的字符没有缓冲。

  3. 为了获得最高效率,可考虑将 OutputStreamWriter 包装到 BufferedWriter 中,以避免频繁调用转换器。例如:

    1
    
    Writer out = new BufferedWriter(new OutputStreamWriter(System.out));//重要
    

    利用标准输出流将文本输出到命令行

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    
    package com.eh.eden.pattern;
       
    import java.io.*;
       
    /**
     * 使用标准输入流,读取键盘录入的数据,存储到临时目录下的a.txt中
     * 将字节输入流转换成字符输入流,InputStreamReader
     */
    class InputStreamReaderDemo {
        public static void main(String[] args) throws IOException {
            try (
                    // //创建输入流对象
                    BufferedReader br = new BufferedReader(new FileReader("/tmp/com.eh/a.txt"));
                    //创建输出流对象
                    BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(System.out))
            ) {
                String line;//用于接收读到的数据
                while ((line = br.readLine()) != null) {
                    bw.write(line);
                    bw.write("\r\n");
                }
            }
        }
    }
    

    使用指定字符集输入和输出

     1
     2
     3
     4
     5
     6
     7
     8
     9
    10
    11
    12
    13
    14
    15
    16
    17
    18
    19
    20
    21
    22
    23
    24
    25
    26
    27
    28
    
    package com.eh.eden.pattern;
       
    import java.io.*;
       
    class TransStreamDemo {
        public static void main(String[] args) throws IOException {
    //        readCN();
            writeCN();
        }
       
        public static void readCN() throws IOException {
            try (
                    //InputStreamReader将字节数组使用指定的编码表解码成文字
                    BufferedReader br = new BufferedReader(new InputStreamReader(new FileInputStream("/tmp/com.eh/a.txt"), "utf-8"))
            ) {
                System.out.println(br.readLine());
            }
        }
       
        public static void writeCN() throws IOException {
            try (
                    OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("/tmp/com.eh/b.txt"), "utf-8")
            ) {
                osw.write("你好");
                osw.write("\r\n");
            }
        }
    }
    

打印流PrintWriter和PrintStream

PrintWriter

1
public class PrintWriter extends Writer
  1. 向文本输出流打印对象的格式化表示形式。此类实现在 PrintStream 中的所有 print 方法。不能输出字节,但是可以输出其他任意类型。
  2. 与 PrintStream 类不同,如果启用了自动刷新,则只有在调用 println、printf 或 format 的其中一个方法时才可能完成此操作,而不是每当正好输出换行符时才完成。这些方法使用平台自有的行分隔符概念,而不是换行符。
  3. 此类中的方法不会抛出 I/O 异常,尽管其某些构造方法可能抛出异常。客户端可能会查询调用 checkError() 是否出现错误。

示例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
package com.eh.eden.pattern;

import java.io.*;

class IODemo {
    public static void main(String[] args) throws IOException {
        try (
                // 创建FileWriter对象时boolean参数表示是否追加;
                // 而创建打印流对象时boolean参数表示是否自动刷新
                PrintWriter pw = new PrintWriter(new FileWriter("/tmp/com.eh/print.txt", true), true)
        ) {
            pw.write("测试打印流");
            pw.println("此句之后换行");
            pw.println("特有功能:自动换行和自动刷新");
            pw.println("利用构造器设置自动刷新");
        }
    }
}

使用字符打印流复制文本文件

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
package com.eh.eden.pattern;

import java.io.*;

class IODemo {
    public static void main(String[] args) throws IOException {
        try (
                BufferedReader br = new BufferedReader(new FileReader("/tmp/com.eh/copyfrom.txt"));
                PrintWriter pw=new PrintWriter("/tmp/com.eh/copyto.txt") {
        }
        ) {
            String line;
            while ((line = br.readLine()) != null) {
                pw.println(line);
            }
        }
    }
}

PrintStream

1
public class PrintStream extends FilterOutputStream implements Appendable, Closeable
  1. PrintStream 为其他输出流添加了功能(增加了很多打印方法),使它们能够方便地打印各种数据值表示形式(例如:希望写一个整数,到目的地整数的表现形式不变。
  2. 与其他输出流不同,PrintStream 永远不会抛出 IOException;而是,异常情况仅设置可通过 checkError 方法测试的内部标志。 另外,为了自动刷新,可以创建一个 PrintStream;这意味着可在写入 byte 数组之后自动调用 flush 方法,可调用其中一个 println 方法,或写入一个换行符或字节 ('\n')
  3. PrintStream 打印的所有字符都使用平台的默认字符编码转换为字节。在需要写入字符而不是写入字节的情况下,应该使用 PrintWriter 类。

使用字节打印流复制文本文件

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
package com.eh.eden.pattern;

import java.io.*;

class IODemo {
    public static void main(String[] args) throws IOException {
        try (
                BufferedReader br = new BufferedReader(new FileReader("/tmp/com.eh/copyfrom.txt"));
                PrintStream ps = new PrintStream("/tmp/com.eh/copyto.txt") {
                }
        ) {
            String line;
            while ((line = br.readLine()) != null) {
                ps.println(line);
            }
        }
    }
}

对象操作流ObjectInputStream和ObjectOutputStream

ObjectInputStream

  1. ObjectOutputStream 将 Java 对象的基本数据类型和图形写入 OutputStream。只能使用 ObjectInputStream 读取(重构)对象。
  2. 只能将支持 java.io.Serializable 接口的对象写入流中。
  3. writeObject 方法用于将对象写入流中。所有对象(包括 String 和数组)都可以通过 writeObject 写入。可将多个对象或基元写入流中。必须使用与写入对象时相同的类型和顺序从相应 ObjectInputstream 中读回对象。

构造方法:

  1. ObjectOutputStream(OutputStream out)

    创建写入指定 OutputStream 的 ObjectOutputStream。

1
public class ObjectOutputStream extends OutputStream implements ObjectOutput, ObjectStreamConstants

ObjectOutputStream

1
public class ObjectInputStream extends InputStream implements ObjectInput, ObjectStreamConstants
  1. ObjectInputStream 对以前使用 ObjectOutputStream 写入的基本数据和对象进行反序列化。
  2. 只有支持 java.io.Serializable 或 java.io.Externalizable 接口的对象才能从流读取。
  3. readObject 方法用于从流读取对象。应该使用 Java 的安全强制转换来获取所需的类型。在 Java 中,字符串和数组都是对象,所以在序列化期间将其视为对象。读取时,需要将其强制转换为期望的类型。

对象读写

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
package com.eh.eden.pattern;

import com.google.common.collect.Lists;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.ToString;

import java.io.*;
import java.util.List;

/**
 * 使用对象输出流写对象和对象输入流读对象
 * 注意:如果Student没有序列化,会抛出java.io.NotSerializableException
 * Serializable:序列号,是一个标识接口,只起标识作用,没有方法
 * 当一个类的对象需要IO流进行读写的时候,这个类必须实现接口
 */
class IODemo {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        writeObjects(Lists.newArrayList(new Student("小明", 13), new Student("小红", 12)));
        List<Student> students = readObjects();
        System.out.println(students);
    }

    public static void writeObject(Student student) throws IOException {
        try (
                // 创建对象输出流的对象
                ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("/tmp/com.eh/a.txt"))
        ) {
            oos.writeObject(student);
        }
    }

    public static Student readObject() throws IOException, ClassNotFoundException {
        try (
                //创建对象输入流的对象
                ObjectInputStream ois = new ObjectInputStream(new FileInputStream("/tmp/com.eh/a.txt"))
        ) {
            Student student = (Student) ois.readObject();
            return student;
        }
    }

    public static void writeObjects(List<Student> students) throws IOException {
        try (
                // 创建对象输出流的对象
                ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("/tmp/com.eh/a.txt"))
        ) {
            oos.writeObject(students);
        }
    }

    public static List<Student> readObjects() throws IOException, ClassNotFoundException {
        List<Student> result = Lists.newArrayList();
        try (
                //创建对象输入流的对象
                ObjectInputStream ois = new ObjectInputStream(new FileInputStream("/tmp/com.eh/a.txt"))
        ) {
            List<Student> students = (List<Student>) ois.readObject();
            return students;
        }
    }
}

@Data
@ToString
@AllArgsConstructor
class Student implements Serializable {
    private static final long serialVersionUID = 4072339518540546994L;

    String name;
    int age;
}

序列化接口Serializable的作用:没有方法,不需要覆写,是一个标记接口为了启动一个序列化功能。唯一的作用就是给每一个需要序列化的类都分配一个序列版本号,这个版本号和类相关联。在序列化时,会将这个序列号也一同保存在文件中,在反序列化时会读取这个序列号和本类的序列号进行匹配,如果不匹配会抛出java.io.InvalidClassException.

注意:静态数据不会被序列化,因为静态数据在方法区,不在对象里。或者使用transient关键字修饰,也不会序列化。

SequenceInputStream

表示其他输入流的逻辑串联。它从输入流的有序集合开始,并从第一个输入流开始读取,直到到达文件末尾,接着从第二个输入流读取,依次类推,直到到达包含的最后一个输入流的文件末尾为止。

案例:媒体文件切割与合并

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
package com.eh.eden.pattern;


import java.io.*;
import java.util.*;

class FileUtils {

    public static void main(String[] args) throws IOException {
        // 切割文件
//        splitFile(new File("/tmp/com.eh/cut.gif"), new File("/tmp/com.eh/cutFiles"));
        // 合并文件
        merge(new File("/tmp/com.eh/cutFiles"));
    }


    /**
     * 将一个媒体文件切割成碎片
     * 思路:1、读取源文件,将源文件的数据分别复制到多个文件
     * 2、切割方式有两种:按照碎片个数切,或者按照指定大小切
     * 3、一个输入流对应多个输出流
     * 4、每个碎片都需要编号,顺序不能错
     */
    private static void splitFile(File srcFile, File partsDir) throws IOException {
        if (!(srcFile.exists() && srcFile.isFile())) {
            throw new RuntimeException("源文件不是正确文件或者不存在");
        }

        if (!partsDir.exists()) {
            partsDir.mkdirs();
        }

        FileOutputStream fos;
        int count = 1; // 文件个数
        try (
                FileInputStream fis = new FileInputStream(srcFile);
        ) {
            byte[] buf = new byte[1024 * 60];
            int len;
            while ((len = fis.read(buf)) != -1) {
                //存储碎片文件
                fos = new FileOutputStream(new File(partsDir, (count++) + ".part"));
                fos.write(buf, 0, len);
                fos.close();
            }
        }

        /*将源文件和切割的信息也保存起来,随着碎片文件一起发送
         * 信息:源文件的名称
         * 碎片的个数
         *将这些信息单独封装到一个文件中
         *还要一个输出流完成此操作 */

//        String fileName = srcFile.getName();
        int partCount = count;
        fos = new FileOutputStream(new File(partsDir, count + ".properties"));
//        fos.write(("fileName="+fileName+System.lineSeparator()).getBytes());
//        fos.write(("fileCount="+Integer.toString(partCount)).getBytes());
        Properties prop = new Properties();
        prop.setProperty("fileName", srcFile.getName());
        prop.setProperty("partCount", Integer.toString(partCount));
        //将属性集中的信息持久化
        prop.store(fos, "part file info");
        fos.close();
    }


    /**
     * 解析配置文件
     *
     * @param configFile
     * @return
     * @throws IOException
     */
    private static Properties getProperties(File configFile) throws IOException {
        FileInputStream fis = null;
        Properties prop;
        try {
            //读取流和配置文件相关联
            fis = new FileInputStream(configFile);
            prop = new Properties();
            //流中的数据加载到集合中
            prop.load(fis);
        } finally {
            if (fis != null) {
                fis.close();
            }
        }
        return prop;
    }


    /**
     * 获取配置文件
     *
     * @param pathDir
     * @return
     */
    public static File getConfigFile(File pathDir) {
        //判断是否存在properties文件
        if (!(pathDir.exists() && pathDir.isDirectory())) {
            throw new RuntimeException(pathDir.toString() + "不是有效目录");
        }
        File[] files = pathDir.listFiles(pathname -> pathname.getName().endsWith(".properties"));
        if (files.length != 1) {
            throw new RuntimeException(pathDir.toString() + "properties扩展名的文件不存在或者不唯一");
        }
        File configFile = files[0];
        return configFile;
    }

    public static void merge(File pathDir) throws IOException {
        Properties prop = getProperties(getConfigFile(pathDir));
        String fileName = prop.getProperty("fileName");
        int partCount = Integer.valueOf(prop.getProperty("partCount"));
        List<FileInputStream> list = new ArrayList<>();
        for (int i = 1; i < partCount; i++) {
            list.add(new FileInputStream(pathDir.toString() + "/" + i + ".part"));
        }
        //List自身无法获取Enumeration工具类,到Collection中找
        Enumeration<FileInputStream> en = Collections.enumeration(list);
        SequenceInputStream sis = new SequenceInputStream(en);
        FileOutputStream fos = new FileOutputStream(pathDir.toString() + "/" + fileName);
        byte[] buf = new byte[1024];
        int len;
        while ((len = sis.read(buf)) != -1) {
            fos.write(buf, 0, len);
        }
        fos.close();
        sis.close();
    }
}

用于操作数组和字符串的流对象

ByteArrayInputStream / ByteArrayOutputStream

CharArrayReader  / CharArrayWriter

StringReader  /  StringWriter

关闭这些流都是无效的,因为这些都未调用系统资源,不需要抛IO异常。

示例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package com.eh.eden.pattern;

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;

/**
 * 源和目的都是内存的读写过程
 * 用流的思想操作数组中的数据
 */
class IODemo {
    public static void main(String[] args) {
        //源:内存
        ByteArrayInputStream bis = new ByteArrayInputStream("andhhshad".getBytes());
        //目的:内存
        ByteArrayOutputStream bos = new ByteArrayOutputStream(); //内部有个可自动增长的buffer数组
        //因为都是源和目的都是内存,没有调用底层资源,所以不要关闭,
        // 即使调用了close也没有任何效果,关闭后仍然可使用,不会抛出异常。
        int ch;
        while ((ch = bis.read()) != -1) {
            bos.write(ch);
        }

        System.out.println(bos.toString());
    }
}

RandomAccessFile

https://gitee.com/lienhui68/picStore/raw/master/null/20200827055838.png

示例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
package com.eh.eden.pattern;

import java.io.IOException;
import java.io.RandomAccessFile;

/**
 * 源和目的都是内存的读写过程
 * 用流的思想操作数组中的数据
 */
class IODemo {
    public static void main(String[] args) throws IOException {
        writeFile();
        readFile();
    }

    public static void readFile() throws IOException {
        try (
                RandomAccessFile raf = new RandomAccessFile("/tmp/com.eh/test.txt", "r")
        ) {
            raf.seek(8 * 1);//读第二个人
            byte[] buf = new byte[6];
            raf.read(buf);
            String name = new String(buf);
            System.out.println("name=" + name);

            int age = raf.readInt();
            System.out.println("age=" + age);
        }

    }

    public static void writeFile() throws IOException {
        try (
                //明确要操作的位置,可以多个线程操作同一份文件而不冲突。多线程下载的基本原理。
                // new完raf后写文件会覆盖
                RandomAccessFile raf = new RandomAccessFile("/tmp/com.eh/test.txt", "rw")
        ) {
            // 初始指针0
            raf.write("张三".getBytes()); // utf8 一个中文字符3个字节
            raf.writeInt(20); // 4个字节
            // 现在指针7
            raf.seek(8);//设置指针
            raf.write("李五".getBytes()); // utf8 一个中文字符3个字节
            raf.writeInt(21);
        }
    }
}

File类

File: 文件和目录路径名的抽象表示形式,File类的实例是不可改变的

File类常用功能

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
File: 文件和目录路径名的抽象表示形式File类的实例是不可改变的

 * File类的构造方法
 *             File(String pathname) 将指定的路径名转换成一个File对象
 *             File(String parent,String child) 根据指定的父路径和文件路径创建对象
 *             File(File parent,String child)
 * File类常用功能
 *             创建boolean createNewFile()当指定文件(或文件夹)不存在时创建文件并返回true否则返回false路径不存在则抛出异常
 *                 boolean mkdir()  当指定文件或文件夹不存在时创建文件夹并返回true否则返回false
 *                 boolean mkdirs() :创建指定文件夹所在文件夹目录不存在时则顺道一块创建        
 *             删除boolean delete()删除文件
            注意要删除一个目录需要先删除这个目录下的所有子文件和子目录
 *             获取File getAbsoluteFile()
 *                 File getParentFile()
 *                 String getAbsolutePath()
 *                 String getParent()
 *                 String getPath()
 *                 long lastModified() 
*             判断 boolean exists();
 *                 boolean isAbsolute() 
 *                 boolean isDirectory() 
 *                 boolean isFile() 
 *                 boolean isHidden()    
 *             修改boolean renameTo(File dest) 将当前File对象所指向的路径修改为指定File所指的路径 修改文件名称

案例:打印指定文件(夹)及其所有子目录

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
package com.eh.eden.pattern;

import java.io.File;

/**
 * 源和目的都是内存的读写过程
 * 用流的思想操作数组中的数据
 */
class IODemo {
    public static void main(String[] args) {
        printTree(new File("/tmp"), 0);
    }

    /**
     * @param f
     * @param level 层数
     */
    public static void printTree(File f, int level) {
        for (int j = 0; j < level; j++) {
            System.out.print("\t");
        }
        System.out.println(f.getAbsolutePath());
        if (f.isDirectory()) {
            level++;
            File[] files = f.listFiles();
            for (int i = 0; i < files.length; i++) {
                printTree(files[i], level + 1);
            }
        }
    }
}

File类重要方法之过滤器

1
2
3
4
5
String[] list()
String[] list(FilenameFilter)
File[] listFiles()
File[] listFiles(FilenameFilter)
File[] listFiles(FileFilter filter)

File类的list方法可以获取目录下的各个文件,传入过滤器还能按照特定需求取出需要的文件。下面来看一下过滤器怎么用的。首先看

1
String[] list(FilenameFilter)

查看FilenameFilter源码,发现其实是一个函数式接口:

1
2
3
4
@FunctionalInterface
public interface FilenameFilter {
    boolean accept(File dir, String name);
}

示例

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
package com.eh.eden.pattern;

import java.io.File;
import java.util.Arrays;

/**
 * 源和目的都是内存的读写过程
 * 用流的思想操作数组中的数据
 */
class IODemo {
    public static void main(String[] args) {
        File file = new File("/tmp/com.eh");
        if (file.isDirectory()) {
            String[] list = file.list((File dir, String name) -> name.endsWith(".txt"));
            Arrays.stream(list).forEach(System.out::println);
        }
    }
}
// 运行结果
copyto.txt
copyfrom.txt
b.txt
a.txt
test.txt
print.txt

IO流使用规律总结

明确要操作的数据是数据源还是数据目的地(要读还是要写)

  • 源:

    InputStream  Reader

  • 目的:

    OutputStream  Writer

明确要操作的设备上的数据是字节还是文本

  • 源:

    字节:InputStream

    文本:Reader

  • 目的:

    字节:OutputStream

    文本:Writer

明确数据所在的具体设备

  • 源设备:
    • 硬盘:文件 File开头
    • 内存:数组,字符串
    • 键盘:System.in
    • 网络:Socket
  • 目的设备:
    • 硬盘:文件 File开头
    • 内存:数组,字符串
    • 屏幕:System.out
    • 网络:Socket

明确是否需要额外功能

  • 需要转换

    转换流 InputStreamReader OutputStreamWriter

  • 需要高效

    缓冲流Bufferedxxx

  • 多个源

    序列流 SequenceInputStream

  • 对象序列化

    ObjectInputStream,ObjectOutputStream

  • 保证数据的输出形式

    打印流PrintStream Printwriter

  • 操作基本数据,保证字节原样性

    DataOutputStream,DataInputStream