- 字符流
字符流的底层其实就是字节流
字符流=字节流+字符集
特点
输入流:一次读一个字节,遇到中文时,一次读多个字节
使用场景
对于纯文本文件进行读写操作
- FileReader类
①创建字符输入流对象
构造方法 | 说明 |
public FileReader(File file) | 创建字符输入流关联本地文件 |
public FileReader(String pathname) | 创建字符输入流关联本地文件 |
如果文件不存在,则直接报错。
②读取数据
成员方法 | 说明 |
public int read() | 读取数据,读到末尾返回-1 |
public int read(char[] buffer) | 读取多个数据,读到末尾返回-1 |
细节1:按字节进行读取。遇到中文,一次读多个字节,读取后解码,返回一个整数。
细节2:读到文件末尾了,read方法返回-1。
③释放资源
成员方法 | 说明 |
public int close() | 释放资源/关流 |
一次读取一个字符:
import java.io.FileReader;
import java.io.IOException;public class FileReadDemo1 {public static void main(String[] args) throws IOException {// 文件FileReader fr = new FileReader("text/a.txt");// read()细节:// 1.read():默认也是一个字节一个字节的读取的,如果遇到中文就会一次读取多个// 2.在读取之后,方法的底层还会进行解码并转成十进制。// 最终把这个十进制作为返回值// 这个十进制的数据也表示在字符集上的数字// 英文:文件里面二进制数据01100001// read方法进行读取,解码并转成十进制97// 中文:文件里面的二进制数据111001101011000110001001// read方法进行读取,解码并转成十进制27721// 我想看到中文汉字,就是把这些十进制数据,再进行强转就可以了// 缓存区,一次读取10个字符char[] buffer = new char[20];// 循环读取for (int r = fr.read(buffer); r != -1; r = fr.read(buffer)){System.out.print(new String(buffer, 0, r));}// 关闭资源fr.close();}
}
读取到缓冲数组中:
import java.io.FileReader;
import java.io.IOException;
import java.nio.charset.StandardCharsets;public class FileReadDemo2 {public static void main(String[] args) throws IOException {// 文件FileReader fr = new FileReader("text/a.txt", StandardCharsets.UTF_8);// read(chars):读取数据,解码,强转三步合并了,// 把强转之后的字符放到数组当中// 空参的read +强转类型转换for (int r = fr.read(); r != -1; r = fr.read()){System.out.println((char) r);}// 关闭资源fr.close();}
}
- FileWriter类
- 构造方法
构造方法 | 说明 |
public Filewriter(File file) | 创建字符输出流关联本地文件 |
public Filewriter(String pathname) | 创建字符输出流关联本地文件 |
public Filewriter(File file,boolean append) | 创建字符输出流关联本地文件,续写 |
public Filewriter(String pathname, boolean append) | 创建字符输出流关联本地文件,续写 |
- 成员方法
成员方法 | 说明 |
void write(int c) | 写出一个字符 |
void write(String str) | 写出一个字符串 |
void write(String str,int off,int len) | 写出一个字符串的一部分 |
void write(char[] cbuf) | 写出一个字符数组 |
void write(char[] cbuf,int off,int len) | 写出字符数组的一部分 |
- FileWriter书写细节
①创建字符输出流对象
细节1:参数是字符串表示的路径或者File对象都是可以的
细节2:如果文件不存在会创建一个新的文件,但是要保证父级路径是存在的
细节3:如果文件已经存在,则会清空文件,如果不想清空可以打开续写开关
②写数据
细节:如果write方法的参数是整数,但是实际上写到本地文件中的是整数在字符集上对应的字符
③释放资源
细节:每次使用完流之后都要释放资源
import java.io.FileWriter;
import java.io.IOException;public class WriterDemo1 {public static void main(String[] args) throws IOException {/* 字节流输出方式// 1 文件输出流FileOutputStream fos = new FileOutputStream("text/b.txt", true);// 2 获取数组byte[] bytes = "\n续写内容".getBytes();fos.write(bytes);// 3 关闭资源fos.close();*/// 1.创建文件写出对象FileWriter fw = new FileWriter("text/b.txt", true);// 2.写出字符//fw.write(97); // 输出一个字符a//fw.write("\nFileWriter续写的内容"); // 输出字符串fw.write("\nFileWriter续写的内容", 0, 11); // 输出部分String内容// 3.续写内容fw.close();}
}
- 字符流原理解析
- 字符输入流原理
①创建字符输入流对象
底层:关联文件,并创建缓冲区(长度为8192的字节数组)
②读取数据
底层:
1.判断缓冲区中是否有数据可以读取
2.缓冲区没有数据:就从文件中获取数据,装到缓冲区中,每次尽可能装满缓冲区如果文件中也没有数据了,返回-1
3.缓冲区有数据:就从缓冲区中读取。
- 空参的read方法:一次读取一个字节,遇到中文一次读多个字节,把字节解码并转成十进制返回
- 有参的read方法:把读取字节,解码,强转三步合并了,强转之后的字符放到数组中
import java.io.FileReader;
import java.io.FileWriter;
import java.io.IOException;public class CharStreamDemo1 {public static void main(String[] args) throws IOException {// 读取器FileReader fr = new FileReader("files\\a.txt");fr.read();// 写出器// 本Writer会清空文件的内容FileWriter fw = new FileWriter("files\\a.txt");// 这里使用fr进行读取// 是否还会取到数据?// 答案是可以,因为缓冲区中有8192字节的数据。// 当缓冲区中数据读取完毕后,无法再读取文件中的内容了。int ch;while ((ch = fr.read()) != -1){System.out.println((char) ch);}// 关闭资源fr.close();fw.close();}
}
- 字符输出流原理
字符输出流也会创建一个8192字节的缓冲区,输出数据先是往缓冲区内写入数据,有3种情况下会将缓冲区中的数据输出到文件中:
- 缓冲区满
- 调用flush方法
- 调用close方法关闭流时
- flush和close方法
成员方法 | 说明 |
public void flush() | 将缓冲区中的数据,刷新到本地文件中 |
public void close() | 释放资源/关流 |
- flush刷新:刷新之后,还可以继续往文件中写出数据
- close关流:断开通道,无法再往文件中写出数据
- 字节流和字符流的使用场景
字节流
拷贝任意类型的文件
字符流
读取纯文本文件中的数据
往纯文本文件中写出数据