IO流
一、流的分类
- 操作数据单位:字节流、字符流
- 数据的流向:输入流、输出流
- 流的角色:节点流、处理流
二、流的体系结构
抽象基类 | 节点流(或文件流) | 缓冲流(处理流的一种) |
---|---|---|
InputStream | FileInputStream | BufferedInputStream |
OutputStream | FileOutputStream | BufferedOutputStream |
Reader | FileReader | BufferedReader |
Writer | FileWriter | BufferedWriter |
说明点:
- read():返回读入的一个字符。如果达到文件末尾,返回-1
- 异常的处理:为了保证流资源一定可以执行关闭操作。需要使用try-catch-finally处理
- 读入的文件一定要存在,否则就会报FileNotFoundException。
将hello.txt文件内容读入程序中,并输入到控制台
@Test
public void test9(){
FileReader fr = null;
try {
// 1.实例化File类的对象,指明要操作的文件
File file = new File("hello.txt");
// 2.提供具体的流
fr = new FileReader(file);
// 3.数据的读入
// read():返回读入的一个字符。如果达到文件末尾,返回-1
// 方式一:
// int data = fr.read();
// while(data != -1) {
// System.out.print((char)data);
// data = fr.read();
// }
// 方式二:语法上针对于方式一的修改
int data;
while((data = fr.read()) != -1 ) {
System.out.print((char)data);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
// 4.流的关闭操作
try {
if(fr != null) {
fr.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
对read()操作升级,使用read的重载方法
@Test
public void test49() {
FileReader fr = null;
try {
// 1.File类的实例化
File file = new File("hello.txt");
// 2.FileReader流的实例化
fr = new FileReader(file);
// 3.读入的操作
// read(char[] cbuf):返回每次读入cbuf数组中的字符的个数。如果达到文件末尾,返回-1
char[] cbuf = new char[5];
int len;
while((len = fr.read(cbuf)) != -1) {
// 方式一:错误的写法
// for (int i = 0; i < cbuf.length; i++) {
// System.out.print(cbuf[i]);
// }
// 正确的写法
// for (int i = 0; i < len; i++) {
// System.out.print(cbuf[i]);
// }
//方式二:错误的写法,对应着方式一的错误的写法
// String str = new String(cbuf);
// System.out.print(str);
// 正确的写法
String str = new String(cbuf,0,len);
System.out.print(str);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
// 4.资源的关闭
try {
if(fr != null) {
fr.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
结合
说明:不能使用字符流来处理图片等字节数据
@Test
public void test113() {
FileReader fr = null;
FileWriter fw = null;
try {
// 1.创建File类的对象,指明读入和写出的文件
File srcFile = new File("hello.txt");
File destFile = new File("hello2.txt");
// 2.创建输入流和输出流的对象
fr = new FileReader(srcFile);
fw = new FileWriter(destFile);
// 3.数据的读入和写出操作
char[] cbuf = new char[5];
int len; // 记录每次读入到cbuf数组中的字符的个数
while((len = fr.read(cbuf)) != -1) {
// 每次写出len个字符
fw.write(cbuf,0,len);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
// 4.关闭流资源
// 方式一:
// try {
// fw.close();
// } catch (IOException e) {
// e.printStackTrace();
// } finally {
// try {
// fr.close();
// } catch (IOException e) {
// e.printStackTrace();
// }
// }
// 方式二:
try {
fw.close();
} catch (IOException e) {
e.printStackTrace();
}
try {
fr.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
从内存中写出数据到硬盘的文件里
说明:
-
输出操作,对应的File可以不存在,并不会报异常
-
File对应的硬盘中的文件如果不存在,在输入的过程中,会自动创建此文件。
File对应的硬盘中的文件如果存在:
如果流使用的构造器是:FileWriter(file,false) / FileWriter(file):对原有文件的覆盖
如果流使用的构造器是:FileWriter(file,true):不会对原有文件覆盖,而是在原有文件基础上追加内容
测试FileInputStream和FileOutputStream的使用
结论:
- 对于文本文件(.txt,. java, .c , .cpp),使用字符流处理
- 对于非文本文件(.jpg, .mp3, .mp4, .avi, .doc, .ppt),使用字节流处理
使用字节流FileInputStream处理文本文件,可能出现乱码。
@Test
public void test5() {
FileInputStream fis = null;
try {
// 1.造文件
File file = new File("hello.txt");
// 2.造流
fis = new FileInputStream(file);
// 3.读数据
byte[] buffer = new byte[5];
int len;
while((len = fis.read(buffer)) != -1) {
String str = new String(buffer,0,len);
System.out.print(str);
}
} catch (IOException e) {
e.printStackTrace();
} finally {
// 4.关闭资源
try {
if(fis != null)
fis.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}