Core Java II—2.输入与输出
输入/输出
读写字节
流家族
OutputStream的结构
OutputStream是一个abstract class
- ByteArrayOutputStream 以字节为单位将数据写入到字节数组中
- FileOutputStream 从文件中以字节的方式读取以字节为单位将数据写入到文件中
- FilterOutputStream 用来封装其他输出流,并为其提供功能
- ZipOutputStream 以字节为单位 向zip文件中写入数据
- PrintStream 装饰其他输出流,可以很方便的打印出各种数据格式的数据
- DataOutputStream 允许程序写入与机器无关的Java数据类型
- BufferedOutputStream 写缓冲流
- ObjectOutputStream 对象序列化,并且用来写出序列化后的数据
- PipedOutputStream 管道输出流
组合输入/输出流过滤器
FilterInputStream
/FilterOutputStream
FileInputStream
/FileOutputStream
与DataInputStream/DataOutputStream
FileInputStream和FileOutputStream可以提供附着在一个磁盘文件上的输入流和输出流。使用name字符串或file对象指定路径名。
与抽象类InputStream和OutputStream一样,这些类只支持在字节级别上的读写:
var fin=new FileInputStream("employee.dat");
byte b=(byte) fin.read();
Java巧妙得把从文件读取数据得方法和读取数值类型得方法分开:
- 某些输入流(如FileInputStream和由URL类得openStream方法返回的输入流)可以从文件和其他更外部的位置上获取字节;
- 其他的输入流(如DataInputStream)可以将字节组装到更有用的数据类型中;
为了从文件中读入数字,首先需要创建一个FileInputStream,然后将其传递给DataInputStream的构造器:
var fin=new FileInputStream("employee.dat");
var din=new DataInputStream(fin);
double x=din.readDouble();
子类可以向处理字节的输入/输出流添加额外的功能。 通过嵌套过滤器来添加多重功能:
BufferedInputStream(InputStream in)
创建一个带缓冲区的输入流。带缓冲区的输入流在从流中读入字符串时,不会每次都访问设备。当缓冲区为空时,会向缓冲区中读入一个新的数据块。
#输入流在默认情况下是不被缓存区缓存的。也就是每个对read的调用都会请求操作系统在分发一个字节。
#相比较之下,请求数据块并将其置于缓冲区中会显得更加高效:
var din=new DataInputStream(new BufferedInputStream(new FileInputStream("employee.dat)));
#使用带缓冲机制的read方法
PushBackInputStream(InputStream in)
构建一个可以预览一个字节或者具有指定尺寸的回推缓冲区的输入流。仅有读入和推回方法。
如果希望能够预先浏览并且读入数字,那么就需要一个既是可推回输入流又是一个数据输入流的引用:
文本输入与输出
var din=new DataInputStream(bpin=new PushbackInputStream(
new BufferedInputStream(
new FileInputStream("employee.dat"))));
如何写出文本输出
使用PrintWriter。为了打印文件,需要用文件名和字符编码方式构建一个PrintStream对象:
var out=new PrintWriter("employee.txt",StandardCharsets.UTF_8);
String name="Harry Hacker";
double salary=75000;
out.print(name);
out.print(' ');
out.println(salary);
程序将Harry Hacker 75000.0输出到写出器out,之后这些字符将会转换成字节并最终写入employee.txt中;
如何读入文本输入
#将短小的文本文件读入到一个字符串中:
var content=(Files.read String Path,charset);
#将文件一行行读入:
List<String> lines=Files.readAllLines(path,charset);
#文件太大,惰性处理为一个Stream<String>对象:
try(Stream<String> lines=Files.lines(path,charset))
{
...
}
#使用扫描器读入符号token,默认分隔符使空白,可以将分隔符修改为任意的正则表达式:
Scanner in=...;
in.useDelimiter("\\PL+"); //接收任何非Unicode字母作为分隔符
#获得包含所有符号的流:
Stream<String> words=in.tokens();