在计算机领域中百分之九十以上的程序拥有着和外部设备交互的功能,这就是我们常说的IO(Input/Output:输入/输出),所谓输入就是外部数据导入计算机内存中的过程,输出则是将内存或者说程序中的数据导入到外部存储中,如数据库、文件以及其他本地磁盘等。
这种输入输出往往遵循着先入先出,顺序存取的特点,像水流一般,因此我们称这样的操作为流(Stream),如下我们根据不同的标准,将IO流分为几个门类:
根据数据流向:
根据处理单位:
根据功能点:
在java.io包中多达40多个类,它们的基类来源于InputStream、OutputStream、Reader、Writer这四个,我们一一看过。
InputStream作为所有字节输入流的父类,主要作用是将外部数据读取到内存中,主要方法如下(JDK8):
我们使用FileInputStream(文件字节输入流)进行如上方法的使用测试:
public class Test {
public static void main(String[] args) throws IOException {
try (InputStream fis = new FileInputStream("E:\\input.txt")) {
System.out.println("可读取字节数:"
+ fis.available());
int content;
long skip = fis.skip(3);
System.out.println("忽略字节数:" + skip);
System.out.print("剩余全量字节:");
while ((content = fis.read()) != -1) {
System.out.print((char) content);
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
输出:
可读取字节数:20
忽略字节数:3
剩余全量字节:name is JavaBuild
outputstream作为所有字节输出流的父类,主要则是将内存或者说程序中的数据以字节流的方式导入到外部存储中,如数据库、文件以及其他本地磁盘等。它的使用方法相比较字节输入流要少:
我们同样以FileOutputStream为例进行上述方法的测试:
public class Test {
public static void main(String[] args) throws IOException {
try (FileOutputStream output = new FileOutputStream("E://output.txt")) {
byte[] array = "JavaBuild".getBytes();
//将一个字节数组写入本地E盘的外部文件output.txt中
output.write(array);
//换行方式1:Windows下的换行符为"\r\n"
output.write("\r\n".getBytes());
//换行方式2:推荐使用,具有良好的跨平台性
String newLine = System.getProperty("line.separator");
output.write(newLine.getBytes());
//输出字节,这里的数字会被转为asicc码中对应的字符
output.write(64);
output.write(56);
output.write(56);
output.write(56);
//关闭输出流
output.close();
} catch (IOException e) {
e.printStackTrace();
}
}
}
效果:
这里可以直接输出单字节数据,也可以输出指定的字节数组。输出字节时以int类型输出,最终根据ASCII表转为字符。如十进制64的转为@符号。
在讲解字符流之前,我们来解释一个面试问题:“为什么有了字节流了还需要使用更耗时的字符流”
确实,字节作为信息存储的最小单元,我们可以通过字节流实现所有信息的输入与输出,但有时候会存在一些问题,比如中文输入时的编码问题,将上述3.1中的测试代码稍微改一下,执行结果如下,中文在控制台输出时乱码了。当然我们可以通过设置编码来规避这个问题,但有时候不晓得编码时,乱码真的会带来潜在风险!
字符流与字节流的区别:
说了这么多,我们现在来看一下Reader这个字符输入流提供的主要方法吧,其实和InputStream差不多,只不过一个是以字节为单位的读取,一个是以字符为单位。
我们将上述3.1中的测试代码稍作加工,采用FileReader流进行输入,打印结果:
可以看到即便有中文,输出在控制台也没有乱码,因为字符流默认采用的是 Unicode 编码。
那么字符流是如何实现txt文件读取的呢?通过FileReader类的继承关系我们可以看到它继承了InputStreamReader,这是一个字节转字符输入流,所以说从根本上,字符流底层依赖的还是字节流!
// 字节流转换为字符流的桥梁
public class InputStreamReader extends Reader {
}
// 用于读取字符文件
public class FileReader extends InputStreamReader {
}
writer是将内存或者说程序中的数据以字符流的方式导入到外部存储中,如数据库、文件以及其他本地磁盘等。
常用方法也和OutputStream相似:
我们同样以FileWriter为例,去测试一下:
public class Test {
public static void main(String[] args) throws IOException {
try (FileWriter fw = new FileWriter("E:\\outwriter.txt")) {
fw.write("大家好!!!");
fw.append("我是JavaBuild");
} catch (IOException e) {
e.printStackTrace();
}
}
}
如果本篇博客对您有一定的帮助,大家记得留言+点赞+收藏呀。原创不易,转载请联系Build哥!
如果您想与Build哥的关系更近一步,还可以关注“JavaBuild888”,在这里除了看到《Java成长计划》系列博文,还有提升工作效率的小笔记、读书心得、大厂面经、人生感悟等等,欢迎您的加入!