一、Java的I/O
1、什么是I/O?
在生活中,你需要将U盘的文件 ,拷贝到电脑(或者将电脑的文件拷贝到其他设备), 文件是通过数据流的方式依次到达另一个设备中, 文件的拷贝就是一个输入(Input)和输出(Output)的过程
Java中提供对应的API支持对文件的输入和输出 , java.io.*
2、什么流?
生活中 也存在流的概念,例如 管道中的流水,从管道的入口到达管道出口,一滴水可以从入口流到出口,可以将“水”比作 “字节数据或字符数据”,数据也可以从一端流到另一端。
输入(Input): Java中,以“应用程序(内存)”为中心,将磁盘文件(设备端)到达内存中的过程 称为 输入
输出(Output): 以“应用程序(内存)”为中心,将数据从内存到达磁盘文件(设备端)的过程称为 输出
根据文件操作方式不同可以将IO分为三类
1、按照读写方向不同: 输入流(InputStream)和输出流(OutputStream)
2、按照数据类型不同: 字节流(Byte)和字符流(Char)
3、按照读写效率不同: 单一流和包装流(buffered等缓冲流)
关于流的分类Java提供 4个顶级抽象类 ,分布构建它们的子类
输入流 | 输出流 | |
---|---|---|
字节流 | InputStream(字节输入流) | OutputStream(字节输出流) |
字符流 | Reader(字符输入流) | Writer(字符的输出流) |
常见的流
1、文件字节输入流和文件字节输出流 : FileInputStream 和 FileOutputStream
2、文件字符输入流和文件字符输出流: FileReader 和 FileWriter
3、缓存字节输入流和 缓存字节输出流 BufferedInputStream 和 BufferedOutputStream
4、缓存字符输入流和缓冲字符输出流 BufferedReader 和BuffereWriter
5、数据输入流和数据输出流: DataInputStream 和 DataOutputStream
6、字节数组输入流 和 字节数组输出流 : ByteArrayInputStream 和 ByteArrayOutputStream
7、字符数组输入流 和 字符数组输出流: CharArrayReader 和 CharArrayWriter
8、转换流: (字节流转成字符流) InputStreamReader 和 OutputStreamWriter
9、对象流(序列化流): ObjectInputStream 和 ObjectOutputStream
10、随机访问流(这个流既可以读,也可以写):RandomAccessFIle
字节流
定义: 文件的输入输出以一个“字节”为单位,进行流处理
FileInputStream 和 FileOutputStream
读入: 将文件中的数据读到内存中
常用方法:
int read() : 一个字节的读取 ,返回字节 的asci码 ,对于汉字会分3次读取
int read(byte) : 按一个数组长度读取, 返回实际读取的字节长度, 数据存放在数组中
int read(byte , offset , len) : 读取流中指定长度的数据,并存放在指定位置,
available() :返回流中剩余的字节长度,如果已读完,则返回0
skip(long n ): 丢弃指定的字节长度,从下一个开始读取
`**available**()`
File file = new File("d:/aaa.txt");
FileInputStream fis = new FileInputStream(file);
//
byte [] b= new byte[10];
StringBuffer sb = new StringBuffer();
//每次读取的长度, b: 存放数据的数组
int len = 0;
while((len = fis.read(b)) !=-1){
sb.append( new String(b,0,len));
}
System.out.println(sb);
fis.close();
public static void read2() throws IOException {
// InputStream是抽象类
InputStream is = new FileInputStream("d:/aaa.txt");
//丢弃前两个字节
is.skip(2);
System.out.println((char)is.read());
System.out.println("还剩下多少个字节:"+ is.available());
// 将后面的字节继续使用字节数组读
byte [] b = new byte[10];
int len = is.read(b,1,4);
// 显示数组中读取的所有数据
System.out.println(Arrays.toString(b));
//将数组的内容转成字符串 对于空内容不会转换
System.out.println(new String(b));
is.close();
}
文件写出: 将内存的数据写出到磁盘中
构造方法:
new FileOutputStream(File/String ) : 构造文件对象的写出流, 默认覆盖写出
new FileOutputStream(File/String , append): 构造文件对象的写出流,
append:表示在原有文件上追加数据, false : 覆盖
常用方法:
void write(int) : 写出一个字节
void writer(byte []) :写出一个字节数组,这里需要指定数组的编码格式 “UTF-8”
void writer(byte[] , offerset,len) : 写出一个字节数组,指定数组的长度和下标。 从数组的下标开始写出,len表示写出长度
flush() :清空缓存,对于使用缓冲流时,将缓冲强制清空。
//将内存的数据写出到文件 如果文件不存在,会自动创建, 默认覆盖写入 true:追加
FileOutputStream fos = new FileOutputStream("d://aaa.txt" ,true);
String str="今天天气还不错";
fos.write(99);
//写出一个字符串 字符串可以转成字节数组 或字符数组
fos.write(str.getBytes("UTF-8"));
// 写出指定长度
fos.write(str.getBytes("UTF-8"),0,3); // 写出这个数组的前2个字节
// 清空缓存
fos.flush();
// 关闭流
fos.close();
System.out.println("写出成功");
文件复制:
将文件(图片,文本,视频)从一个目录复制到另一个目录, 其中数据长度不变,通过文件读写的方式完成复制
复制过程:从源文件读取数据,然后将数据再出到目标文件中。
[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-AY58MXtm-1604883425216)(assets/copy.png)]
/**
* 单个字节复制
* @param srcFile 源文件
* @param disFile 目标文件
*/
public static void copyFile(File srcFile, File disFile){
FileInputStream fis=null;
FileOutputStream fos =null;
try {
// 源文件输入流
fis = new FileInputStream(srcFile);
// 目标文件输出流
fos = new FileOutputStream(disFile);
int n=0;
while( (n =fis.read()) !=-1){
//将读到的n写出到 目标文件中
fos.write(n);
}
System.out.println("复制成功。。");
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally{
//无论是否发生异常 都会关闭流
try {
fos.close();
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
/**
* 一个字节数组的赋值
* @param src 源地址
* @param disc 目标地址
*/
public static void copyFile(String src,String disc){
FileInputStream fis = null;
FileOutputStream fos = null;
try {
//创建 字节输入流
fis=new FileInputStream(src);
fos = new FileOutputStream(disc);
int len=0;
byte [] b = new byte[1024];
while( (len= fis.read(b)) !=-1){
// 写出 实际读取的长度 ,为了避免在最后一次写出时出现多余字节
fos.write(b,0,len);
}
System.out.println("复制成功");
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
} finally{
try {
fos.close();
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
字符流
字符流用于读写存储字符的文件, 以一个字符为单位,一依次读取字符 文件, 常用类以 Reader或Writer为父类, 对文件的操作使用 java.io.FileReader 和java.io.FileWriter
读文件 :FileReader
常用方法:
new FileReader(path): 通过文件路径构建字符输入流
new FileReader(File):通过文件对象构建字符输入流
- int read() :读取一个字符 ,返回字符的int类型
- int read(char ):读取字符数组长度的数据 ,返回实际读取字符长度,数据存放在字符数组中
- int read(char offerset len):读取指定字符长度的数组,返回 实际读取字符的长度,数据存放在字符数组中
- mark(int) :标记流中当前位置 (读取到哪里了)
- markSupported():判断此流是否支持mark操作
- reset(): 重置流数据,(又可从头开始读取)
- skip(long) :丢弃指定长度字符
读字符文件
// 1、创建字符输入流
try {
FileReader reader = new FileReader("d:/myfile.txt");
// 丢弃字符
reader.skip(1);
//读一个字符
System.out.println((char)reader.read());
System.out.println((char)reader.read());
//读一个字符数组长度
char [] c = new char[10];
System.out.println("实际长度:"+reader.read(c));
System.out.println(new String(c));
//继续读
int len = reader.read(c,0,5);
System.out.println("字符数组:"+ Arrays.toString(c));
System.out.println("读指定长度字符个数:"+new String(c,0,len));
// 将字符流重置
// reader.reset();
// System.out.println("重置后继续读:"+ reader.read());
//System.out.println("是否支持标记字符:"+reader.markSupported());
//关闭流,后面就不能使用该对象
reader.close();
} catch (FileNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
写文件: 将内存数据写出到文件中,在写出过程中可以 覆盖写出也可以追加写出,FileWriter类创建对象过程
new FileWriter(String ):指定写出文件地址
new FileWriter(String ,append) : 指定写出文件地址,设置是否追加写出,true表示追加,false表示覆盖
new FileWriter(File)指定写出文件对象
new FileWriter(File ,append);指向写出文件对象,设置是否可追加
常用方法:
writer(int) :写出一个字符
writer(String):写出一个字符串
writer(char [] ):写出一个字符数组
writer(char [] , offerset , len):写出一个指定长度的字符数组
flush() :刷新缓冲,
close():关闭缓冲
append© :将指定字符添加到此流中
// 1、创建文件写出流 FileWriter
try {
// 文件不存在,可自动创建,但是不会创建目录
File file = new File("d://myabc/aaa.txt");
//判断文件目录不存在, 先创建目录
if(!file.getParentFile().exists()){
//创建该目录
file.getParentFile().mkdirs();
}
FileWriter writer = new FileWriter("d://myabc/aaa.txt");
// 写一个字符的asci
writer.write(98);
//写字符串
writer.write("hello");
//写指定长度的字符串
writer.write("abcdef",0,3); //写abc
char [] c = {'L','O','L'};
//写字符数组
writer.write(c);
System.out.println("写出成功");
writer.flush();
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
关闭和刷新:
对于带有缓冲 功能的写出流,需要先刷新缓冲区,才能将数据写出,如果不刷新则最后不能正常写出。写出流如果刷新后还可以继续写,而关闭了则不能继续写。
面试题 flush 和close的区别?
flush: 刷新缓冲 ,流可以继续使用
close: 先刷新缓冲器,然后再释放系统资源, 关闭后不能继续使用
try {
FileWriter writer = new FileWriter("1.txt");
writer.write("刷");
writer.flush();
writer.write("新");
writer.flush();
writer.write("关");
writer.close();
writer.write("闭"); // 这里抛出异常 , Stream closed
writer.close();
} catch (IOException e) {
e.printStackTrace();
}
关于换行符
回车符 \r 和换行符 \n :
回车符:回到一行的开头(return)。
换行符:下一行(newline)。
系统中的换行:
Windows系统里,每行结尾是 回车+换行 ,即 \r\n ;
Unix系统里,每行结尾只有 换行 ,即 \n ;
Mac系统里,每行结尾是 回车 ,即 \r 。从 Mac OS X开始与Linux统一。