学习笔记:IO流

图1. IO流的体系(部分​​​​)

前面所说的File类,并不能访问文件内容本身,要实现 对文件内容的操作(输入/输出)就需要用到 IO流,他可以方便实现数据的 输入/输出 的操作。Java中把不同的 输入/输出源(键盘、文件、网络访问等等)抽象表述为“流”(stream),通过流的方式允许Java程序使用相同的的方式来访问不同的输入/输出源。

java把所有传统的流类型都放在java.io包中。

1. 流的分类

按照不同方式,流的分类方式如 图1 所示,其中,按照输入/输出的方向,都是从内存的角度来看。

Java的输入流主要由InputStream和Reader作为基类,输出流则主要由OutputStream和Writer作为基类。这四个都是抽象类,无法直接实例化。

字节流和字符流用法基本一样,区别就是字节流操作的数据单元是8位的字节,字符流操作的数据单元是16位的字符。

【流按照角色来分:可以分为节点流和处理流。     节点流可以从一个特定的IO设备读/写数据的流,节点流也成为低级流;    处理流则用于对一个已经存在的流进行连接或封装,通过封装后的流实现数据读/写,处理流也称为高级流。】

2. 字节流和字符流

输入流——字节输入流

InputStream 和 Reader是所有输入流的抽象基类,他们的方法是所有输入流都可使用的。

InputStream: 

java.io.InputStream 抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存中。它定义了字节输入流的基本共性功能方法。

void close() :关闭此输入流并释放与此流相关联的任何系统资源。
abstract int read() : 从输入流读取数据的下一个字节。
int read(byte[] b) : 从输入流中读取一些字节数,并将它们存储到字节数组 b中 。

int read(byte[] b, int off,int len):从off开始,将输入流中最多 len 个数据字节读入 byte 数组。

[close方法,当完成流的操作时,必须调用此方法,释放系统资源]

FileInputStream类 :

java.io.FileInputStream 类是文件输入流,从文件中读取字节。是InputStream的子类。

构造方法
FileInputStream(File file) : 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的 File对象 file命名。
FileInputStream(String name) : 通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的路径名 name命名。
创建一个流对象时,必须传入一个文件路径。该路径下,如果没有该文件,会抛出FileNotFoundException

 读取字节数据:

// 使用相对路径称创建流对象
FileInputStream fis = new FileInputStream("read.txt");
// 读取数据,返回一个字节
int read = fis.read();
System.out.println((char) read);
read = fis.read();
System.out.println((char) read);
read = fis.read();
System.out.println((char) read);
read = fis.read();
System.out.println((char) read);
read = fis.read();
System.out.println((char) read);
// 读取到末尾,返回‐1
read = fis.read();
System.out.println( read);
// 关闭资源
fis.close();

/* ******************循环改进读取方式***************** */
FileInputStream fis = new FileInputStream("read.txt");
// 定义变量,保存数据
int b ;
// 循环读取
while ((b = fis.read())!=‐1) {
    System.out.println((char)b);
}
// 关闭资源
fis.close();

/* ******************字节数组读取方式***************** */
// 使用文件名称创建流对象.
FileInputStream fis = new FileInputStream("read.txt"); // 文件中为abcde
// 定义变量,作为有效个数
int len ;
// 定义字节数组,作为装字节数据的容器
byte[] b = new byte[2];
// 循环读取
while (( len= fis.read(b))!=‐1) {
    // 每次读取后,把数组变成字符串打印
    System.out.println(new String(b));
}
// 关闭资源
fis.close();

假设read.txt中内容是 abcde。那么当时用长度为2的字节数组读取时,得到的会是如下结果:

输出结果:
ab
cd
ed

为什么会这样?呢是因为字节数组长度为2,所以每次读取只能装两个,第三次读取时,实际只剩一个字节没有读取,也就是 e ,所以第三次读取完时 e 替换了 第二次读取时的 c ,但是 d 没有字节去替换,所以试 ed。为了避免这种情况,需要改进一下上述字节数组的读取:

FileInputStream fis = new FileInputStream("read.txt"); // 文件中为abcde
int len ;
byte[] b = new byte[2];
while (( len= fis.read(b))!=‐1) {
// 每次读取后,把数组的有效字节部分,变成字符串打印
System.out.println(new String(b,0,len));// len 每次读取的有效字节个数
}
fis.close();

输入流——字符输入流

Reader:

java.io.Reader 抽象类是表示用于读取字符流的所有类的超类,可以读取字节信息到内存中。它定义了字节输入流的基本共性功能方法。

void close() :关闭此流并释放与此流相关联的任何系统资源。
int read() : 从输入流读取一个字符。
int read(char[] cbuf) : 从输入流中读取一些字符,并将它们存储到字符数组 cbuf中 。

int read(char[] b, int off,int len):从off开始,将输入流中最多 len 个数据字节读入 char 数组。

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

FileReader类:

java.io.FileReader 类是读取字符文件的便利类。构造时使用系统默认的字符编码(字节与字符的对应规则)和默认字节缓冲区(一个字节数组,用来临时存储字节数据)

构造方法
FileReader(File file) : 创建一个新的 FileReader ,给定要读取的File对象。
FileReader(String fileName) : 创建一个新的 FileReader ,给定要读取的文件的名称。
创建一个流对象时,必须传入一个文件路径。类似于FileInputStream 。

读取字符数据:

// 使用文件名称创建流对象
//read.txt 中是“我爱你”
FileReader fr = new FileReader("read.txt");
// 定义变量,保存数据
int b ;
// 循环读取
while ((b = fr.read())!=‐1) {
    System.out.println((char)b);
}
// 关闭资源
fr.close();

/* *************使用字符数组读取**************** */
FileReader fr = new FileReader("read.txt");
int len ;
// 定义字符数组,作为装字符数据的容器
char[] cbuf = new char[2];
while ((len = fr.read(cbuf))!=‐1) {
    System.out.println(new String(cbuf,0,len));
}
fr.close();

 输出流——字节输出流

OutputStream

java.io.OutputStream 抽象类是表示字节输出流的所有类的超类,将指定的字节信息写出到目的地。它定义了字节输出流的基本共性功能方法。

void close() :关闭此输出流并释放与此流相关联的任何系统资源。
void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
void write(byte[] b) :将 b.length字节从指定的字节数组写入此输出流。
void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。
abstract void write(int b) :将指定的字节输出流。

FileOutputStream类

java.io.FileOutputStream 类是文件输出流,用于将数据写出到文件。是OutputStream众多子类之一。

构造方法
FileOutputStream(File file) :创建文件输出流以写入由指定的 File对象表示的文件。
FileOutputStream(String name) : 创建文件输出流以指定的名称写入文件。
当你创建一个流对象时,必须传入一个文件路径。该路径下,如果没有这个文件,会创建该文件。如果有这个文件,会清空这个文件的数据。

写出字节数据:

// 使用文件名称创建流对象
FileOutputStream fos = new FileOutputStream("fos.txt");
// 写出数据
fos.write(97); // 写出第1个字节
fos.write(98); // 写出第2个字节
fos.write(99); // 写出第3个字节
// 关闭资源
fos.close();

/* ****************写出字节数组***************** */
FileOutputStream fos = new FileOutputStream("fos.txt");
// 字符串转换为字节数组
byte[] b = "我爱中国".getBytes();
// 写出字节数组数据
fos.write(b);
// 关闭资源
fos.close();

经过以上的演示,每次程序运行,创建输出流对象,都会清空目标文件中的数据。如何保留目标文件中数据,还能
继续添加新数据呢?

FileOutputStream(File file, boolean append) : 创建文件输出流以写入由指定的 File对象表示的文件。
FileOutputStream(String name, boolean append) : 创建文件输出流以指定的名称写入文件。
这两个构造方法,参数中都需要传入一个boolean类型的值, true 表示追加数据, false 表示清空原有数据。这样创建的输出流对象,就可以指定是否追加续写了

FileOutputStream fos = new FileOutputStream("fos.txt",true);
byte[] b = "abcde".getBytes();
// 写出从索引2开始,2个字节。索引2是c,两个字节,也就是cd。
fos.write(b);
// 关闭资源
fos.close();

/* 
*    Windows系统里,换行符号是\r\n
*/
FileOutputStream fos = new FileOutputStream("fos.txt");
byte[] words = {97,98,99,100,101};
for (int i = 0; i < words.length; i++) {
fos.write(words[i]);
// 写出一个换行, 换行符号转成数组写出
fos.write("\r\n".getBytes());
}
fos.close();

输出流——字符输出流

java.io.Writer 抽象类是表示用于写出字符流的所有类的超类,将指定的字符信息写出到目的地。它定义了字节输出流的基本共性功能方法

abstract void close() :关闭此输出流并释放与此流相关联的任何系统资源。
abstract void flush() :刷新此输出流并强制任何缓冲的输出字符被写出。
void write(int c) :写出一个字符。
void write(char[] cbuf) :将 b.length字符从指定的字符数组写出此输出流。
abstract void write(char[] b, int off, int len) :从指定的字符数组写出 len字符,从偏移量off开始输出到此输出流。
public void write(String str) :写出一个字符串。

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

FileWriter类

java.io.FileWriter 类是写出字符到文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。

构造方法
FileWriter(File file) : 创建一个新的 FileWriter,给定要读取的File对象。
FileWriter(String fileName) : 创建一个新的 FileWriter,给定要读取的文件的名称。
当你创建一个流对象时,必须传入一个文件路径,类似于FileOutputStream。

// 使用文件名称创建流对象
FileWriter fw = new FileWriter("fw.txt");
// 写出数据
fw.write(97); // 写出第1个字符
fw.write('b'); // 写出第2个字符
fw.write('C'); // 写出第3个字符
fw.write(200013); // 写出第4个字符,中文编码表中30000对应一个汉字。
/*关闭资源时,与FileOutputStream不同。
如果不关闭,数据只是保存到缓冲区,并未保存到文件。*/

因为内置缓冲区的原因,如果不关闭输出流,无法写出字符到文件中。但是关闭的流对象,是无法继续写出数据的。如果我们既想写出数据,又想继续使用流,就需要flush 方法了。

// 使用文件名称创建流对象
FileWriter fw = new FileWriter("fw.txt");
// 写出数据,通过flush
fw.write('刷'); // 写出第1个字符
fw.flush();
fw.write('新'); // 继续写出第2个字符,写出成功
fw.flush();
// 写出数据,通过close
fw.write('关'); // 写出第1个字符
fw.close();
fw.write('闭'); // 继续写出第2个字符,【报错】java.io.IOException: Stream closed
fw.close();

 字节流练习:图片复制

// 1.创建流对象
// 1.1 指定数据源
FileInputStream fis = new FileInputStream("D:\\test.jpg");
// 1.2 指定目的地
FileOutputStream fos = new FileOutputStream("test_copy.jpg");
// 2.读写数据
// 2.1 定义数组
byte[] b = new byte[1024];
// 2.2 定义长度
int len;
// 2.3 循环读取
while ((len = fis.read(b))!=‐1) {
    // 2.4 写出数据
    fos.write(b, 0 , len);
}
// 3.关闭资源
fos.close();
fis.close();

IO异常的处理

JDK7前处理 : 建议使用try...catch...finally 代码块,处理异常部分

// 声明变量
FileWriter fw = null;
try {
    //创建流对象
    fw = new FileWriter("fw.txt");
    // 写出数据
    fw.write("我爱你"); 
} catch (IOException e) {
    e.printStackTrace();
} finally {
    try {
        if (fw != null) {
            fw.close();
        }
    } catch (IOException e) {
        e.printStackTrace();
    }
}

JDK7的处理 : 还可以使用JDK7优化后的try-with-resource 语句,该语句确保了每个资源在语句结束时关闭。所谓的资源
(resource)是指在程序完成后,必须关闭的对象。

         格式: 

try (创建流对象语句,如果多个,使用';'隔开) {
        // 读写数据
} catch (IOException e) {
        e.printStackTrace();
}

// 创建流对象
try ( FileWriter fw = new FileWriter("fw.txt"); ) {
    // 写出数据
    fw.write("黑马程序员"); //黑马程序员
} catch (IOException e) {
    e.printStackTrace();
}

JDK9中try-with-resource 的改进,对于引入对象的方式,支持的更加简洁。被引入的对象,同样可以自动关闭,无需手动close

        格式:

// 被final修饰的对象
final Resource resource1 = new Resource("resource1");
// 普通对象
Resource resource2 = new Resource("resource2");
// 引入方式:直接引入
try (resource1; resource2) {
       // 使用对象
}

// 创建流对象
final FileReader fr = new FileReader("in.txt");
FileWriter fw = new FileWriter("out.txt");
// 引入到try中
try (fr; fw) {
    // 定义变量
    int b;
    // 读取数据
    while ((b = fr.read())!=‐1) {
        // 写出数据
        fw.write(b);
    }
} catch (IOException e) {
    e.printStackTrace();
}


缓冲流

缓冲流,也叫高效流,是对4个基本的FileXxxx 流的增强,所以也是4个流,按照数据类型分类:

字节缓冲流: BufferedInputStream , BufferedOutputStream
字符缓冲流: BufferedReader , BufferedWriter

基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO
次数,从而提高读写的效率。

①字节缓冲流    构造方法 

public BufferedInputStream(InputStream in) :创建一个 新的缓冲输入流。
public BufferedOutputStream(OutputStream out) : 创建一个新的缓冲输出流。

缓冲流读写方法与基本的流是一致的,所以来比较一下效率,来说明缓冲流存在的意义:

//效率测试(可自行选择一个大文件)
/* **************************************************************************** */
// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
    FileInputStream fis = new FileInputStream("jdk9.exe");
    FileOutputStream fos = new FileOutputStream("copy.exe")
){
    // 读写数据
    int b;
    while ((b = fis.read()) != ‐1) {
        fos.write(b);
    }
} catch (IOException e) {
    e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("普通流复制时间:"+(end ‐ start)+" 毫秒");

/* **************************************************************************** */
// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
    BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk9.exe"));
    BufferedOutputStream bos = new BufferedOutputStream(new               FileOutputStream("copy.exe"));
){
    // 读写数据
    int b;
    while ((b = bis.read()) != ‐1) {
        bos.write(b);
    }
} catch (IOException e) {
    e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("缓冲流复制时间:"+(end ‐ start)+" 毫秒");

如何更快呢?呢就是使用使用数组的方式

// 记录开始时间
long start = System.currentTimeMillis();
// 创建流对象
try (
    BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk9.exe"));
    BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.exe"));
){
    // 读写数据
    int len;
    byte[] bytes = new byte[8*1024];
    while ((len = bis.read(bytes)) != ‐1) {
        bos.write(bytes, 0 , len);
    }
} catch (IOException e) {
    e.printStackTrace();
}
// 记录结束时间
long end = System.currentTimeMillis();
System.out.println("缓冲流使用数组复制时间:"+(end ‐ start)+" 毫秒");

②字符缓冲流     构造方法

public BufferedReader(Reader in) :创建一个 新的缓冲输入流。
public BufferedWriter(Writer out) : 创建一个新的缓冲输出流。

特有方法

字符缓冲流的基本方法与普通字符流调用方式一致,不再阐述,我们来看它们具备的特有方法。
BufferedReader: public String readLine() : 读一行文字。
BufferedWriter: public void newLine() : 写一行行分隔符,由系统属性定义符号。

//readLine 方法
// 创建流对象
BufferedReader br = new BufferedReader(new FileReader("in.txt"));
// 定义字符串,保存读取的一行文字
String line = null;
// 循环读取,读取到最后返回null
while ((line = br.readLine())!=null) {
    System.out.print(line);
    System.out.println("‐‐‐‐‐‐");
}
// 释放资源
br.close();

//newLine 方法
// 创建流对象
BufferedWriter bw = new BufferedWriter(new FileWriter("out.txt"));
// 写出数据
bw.write("i");
// 写出换行
bw.newLine();
bw.write("love");
bw.newLine();
bw.write("you");
bw.newLine();
// 释放资源
bw.close();

转换流

☆☆☆☆☆☆转换流是字节与字符间的桥梁!

【在此之前,需要搞清楚的是:字符集和字符编码

编码引出的问题
在IDEA中,使用FileReader 读取项目中的文本文件。由于IDEA的设置,都是默认的UTF-8 编码,所以没有任何问题。但是,当读取Windows系统中创建的文本文件时,由于Windows系统的默认是GBK编码,就会出现乱码。

       FileReader fileReader = new FileReader("E:\\File_GBK.txt");
       int read;
       while ((read = fileReader.read()) != ‐1) {
              System.out.print((char)read);
       }
       fileReader.close();

如何读取GBK编码的文件?

①InputStreamReader类

转换流java.io.InputStreamReader ,是Reader的子类,是从字节流到字符流的桥梁。它读取字节,并使用指定的字符集将其解码为字符。它的字符集可以由名称指定,也可以接受平台的默认字符集。

构造方法
InputStreamReader(InputStream in) : 创建一个使用默认字符集的字符流。
InputStreamReader(InputStream in, String charsetName) : 创建一个指定字符集的字符流。

// 定义文件路径,文件为gbk编码
String FileName = "E:\\file_gbk.txt";
// 创建流对象,默认UTF8编码
InputStreamReader isr = new InputStreamReader(new FileInputStream(FileName));
// 创建流对象,指定GBK编码
InputStreamReader isr2 = new InputStreamReader(new FileInputStream(FileName) , "GBK");
// 定义变量,保存字符
int read;
// 使用默认编码字符流读取,乱码
while ((read = isr.read()) != ‐1) {
    System.out.print((char)read);
}
isr.close();

// 使用指定编码字符流读取,正常解析
while ((read = isr2.read()) != ‐1) {
    System.out.print((char)read);// 大家好
}
isr2.close();

②OutputStreamWriter类

转换流java.io.OutputStreamWriter ,是Writer的子类,是从字符流到字节流的桥梁。使用指定的字符集讲字符编码为字节。它的字符集可以由名称指定,也可以接受平台的默认字符集。

构造方法
OutputStreamWriter(OutputStream in) : 创建一个使用默认字符集的字符流。
OutputStreamWriter(OutputStream in, String charsetName) : 创建一个指定字符集的字符流。

// 定义文件路径
String FileName = "E:\\out.txt";
// 创建流对象,默认UTF8编码
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(FileName));
// 写出数据
osw.write("你好"); // 保存为6个字节
osw.close();
// 定义文件路径
String FileName2 = "E:\\out2.txt";
// 创建流对象,指定GBK编码
OutputStreamWriter osw2 = new OutputStreamWriter(new FileOutputStream(FileName2),"GBK");
// 写出数据
osw2.write("你好");// 保存为4个字节
osw2.close();

序列化

当我们如果想要把自定义的对象保存到文件中,并读取出来,该怎么做?

Java 提供了一种对象序列化的机制。用一个字节序列可以表示一个对象,该字节序列包含该对象的数据、对象的
类型和对象中存储的数据等信息。【将数据结构或对象转换成二进制串的过程】

反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中
存储的数据信息,都可以用来在内存中创建对象【:二进制串转换成数据结构或对象的过程】

ObjectOutputStream

ObjectOutputStream 将 Java 对象的基本数据类型和图形写入 OutputStream。可以使用 ObjectInputStream 读取(重构)对象。通过在流中使用文件可以实现对象的持久存储。

只能将支持 java.io.Serializable 接口的对象写入流中。每个 serializable 对象的类都被编码,编码内容包括类名和类签名、对象的字段值和数组值,以及从初始对象中引用的其他所有对象的闭包。

构造方法
public ObjectOutputStream(OutputStream out) : 创建一个指定OutputStream的ObjectOutputStream

一个对象要想序列化,必须满足两个条件:

①该类必须实现java.io.Serializable 接口, Serializable 是一个标记接口,不实现此接口的类将不会使任何状态序列化或反序列化,会抛出NotSerializableException 。

②该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用
transient 关键字修饰。

public class People implements java.io.Serializable {
    public String name;
    public String address;
    public transient int age; // transient瞬态修饰成员,不会被序列化
    public void addressCheck() {
        System.out.println("Address check : " + name + " ‐‐ " + address);
    }
}

/* *********************写出对象方法*********************** */
People e = new people ();
e.name = "zhangsan";
e.address = "beiqinglu";
e.age = 20;
try {
    // 创建序列化流对象
    ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("people .txt"));
    // 写出对象
    out.writeObject(e);
    // 释放资源
    out.close();
    fileOut.close();
    System.out.println("Serialized data is saved"); // 姓名,地址被序列化,年龄没有被序列
化。
} catch(IOException i) {
    i.printStackTrace();
}

ObjectInputStream

ObjectInputStream 对以前使用 ObjectOutputStream 写入的基本数据和对象进行反序列化。

构造方法
public ObjectInputStream(InputStream in) : 创建一个指定InputStream的ObjectInputStream

Peoplee = null;
try {
    // 创建反序列化流
    FileInputStream fileIn = new FileInputStream("people.txt");
    ObjectInputStream in = new ObjectInputStream(fileIn);
    // 调用ObjectInputStream,读取一个对象
    e = (People) in.readObject();
    // 释放资源
    in.close();
    fileIn.close();
}catch(IOException i) {
    // 捕获其他异常
    i.printStackTrace();
    return;
}catch(ClassNotFoundException c) {
    // 捕获类找不到异常
    System.out.println("Employee class not found");
    c.printStackTrace();
    return;
}
// 无异常,直接打印输出
System.out.println("Name: " + e.name); // zhangsan
System.out.println("Address: " + e.address); // beiqinglu
System.out.println("age: " + e.age); // 0

对于JVM可以反序列化对象,它必须是能够找到class文件的类。如果找不到该类的class文件,则抛出一个ClassNotFoundException 异常。

另外,当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操
作也会失败,抛出一个InvalidClassException 异常

异常的原因:
该类的序列版本号与从流中读取的类描述符的版本号不匹配
该类包含未知数据类型
该类没有可访问的无参数构造方法

Serializable 接口给需要序列化的类,提供了一个序列版本号。serialVersionUID 该版本号的目的在于验证序列化的对象和对应类是否版本匹配。

public class People implements java.io.Serializable {
    // 加入序列版本号
    private static final long serialVersionUID = 1L;
    public String name;
    public String address;
    // 添加新的属性 ,重新编译, 可以反序列化,该属性赋为默认值.
    public int eid;
    public void addressCheck() {
        System.out.println("Address check : " + name + " ‐‐ " + address);
    }
}

打印流

平时我们在控制台打印输出,是调用print 方法和println 方法完成的,这两个方法都来自于java.io.PrintStream 类,该类能够方便地打印各种数据类型的值,是一种便捷的输出方式。

构造方法
public PrintStream(String fileName) : 使用指定的文件名创建一个新的打印流。

改变打印流向
System.out 就是PrintStream 类型的,只不过它的流向是系统规定的,打印在控制台上。不过,既然是流对象,我们就,改变它的流向。

// 调用系统的打印流,控制台直接输出97
System.out.println(97);
// 创建打印流,指定文件的名称
PrintStream ps = new PrintStream("ps.txt");
// 设置系统的打印流流向,输出到ps.txt
System.setOut(ps);
// 调用系统的打印流,ps.txt中输出97
System.out.println(97);

猜你喜欢

转载自blog.csdn.net/weixin_38816084/article/details/82848609