IO流的概念和分类
流是一组有顺序的,有起点和终点的字节集合,是对数据传输的总称或抽象。即数据在两设备间的传输称为流,流的本质是数据传输,根据数据传输特性将流抽象为各种类,方便更直观的进行数据操作。
在IO中涉及的设备文件包括文件、控制台、网络链接等。
IO流的分类
- 根据处理数据类型的不同分为:字符流和字节流;
- 根据数据流向不同分为:输入流和输出流。
字符流和字节流的区别
字符流的由来: 因为数据编码的不同,而有了对字符进行高效操作的流对象。本质其实就是基于字节流读取时,去查了指定的码表。字节流和字符流的区别:
- 读写单位不同:字节流以字节(8bit)为单位,字符流以字符为单位,根据码表映射字符,一次可能读多个字节。
- 处理对象不同:字节流能处理所有类型的数据(如图片、avi等),而字符流只能处理字符类型的数据。
- 使用缓冲区不同:字节流在操作的时候本身是不会用到缓冲区的,是文件本身的直接操作的;而字符流在操作的时候下后是会用到缓冲区的,是通过缓冲区来操作文件的。
结论:优先选用字节流。首先因为硬盘上的所有文件都是以字节的形式进行传输或者保存的,包括图片等内容。但是字符只是在内存中才会形成的,所以在开发中,字节流使用广泛。
IO流类图结构
IO流对象
输入字节流InputStream
InputStream 是所有的输入字节流的父类,它是一个抽象类。
ByteArrayInputStream、StringBufferInputStream、FileInputStream 是三种基本的介质流,它们分别从Byte 数组、StringBuffer、和本地文件中读取数据。
PipedInputStream 是从与其它线程共用的管道中读取数据。
ObjectInputStream 和所有FilterInputStream的子类都是装饰流(装饰器模式的主角)。意思是FileInputStream类可以通过一个String路径名创建一个对象,FileInputStream(String name)。
而DataInputStream必须装饰一个类才能返回一个对象,DataInputStream(InputStream in)。
【案例 】读取文件内容:
public static void main(String[] args) throws IOException {
String fileName="D:"+File.separator+"in.txt";
File f=new File(fileName);
// 通过文件创建文件输入流
InputStream in=new FileInputStream(f);
byte[] b=new byte[(int)f.length()];
in.read(b);// 读取文件流到byte数组中
// in.read();// 一个字节一个字节的读取,返回-1表示文件末尾了
in.close();
System.out.println(new String(b));
}
输出字节流OutputStream
OutputStream 是所有的输出字节流的父类,它是一个抽象类。
ByteArrayOutputStream、FileOutputStream是两种基本的介质流,它们分别向Byte 数组、和本地文件中写入数据。
PipedOutputStream 是向与其它线程共用的管道中写入数据。
ObjectOutputStream 和所有FilterOutputStream的子类都是装饰流。具体例子跟InputStream是对应的。
【案例 】向文件中写入字符串:
public static void main(String[] args) throws IOException {
String fileName="D:"+File.separator+"hello.txt";
File f=new File(fileName);
OutputStream out =new FileOutputStream(f);
String str="Hello World";
byte[] b=str.getBytes();
out.write(b);
out.close();
}
字符输入流Reader
Reader 是所有的输入字符流的父类,它是一个抽象类。
CharReader、StringReader是两种基本的介质流,它们分别将Char 数组、String中读取数据。PipedReader 是从与其它线程共用的管道中读取数据。
BufferedReader 很明显就是一个装饰器,它和其子类负责装饰其它Reader 对象。
FilterReader 是所有自定义具体装饰流的父类,其子类PushbackReader 对Reader 对象进行装饰,会增加一个行号。
InputStreamReader 是一个连接字节流和字符流的桥梁,它将字节流转变为字符流。FileReader可以说是一个达到此功能、常用的工具类,在其源代码中明显使用了将FileInputStream 转变为Reader 的方法。我们可以从这个类中得到一定的技巧。Reader 中各个类的用途和使用方法基本和InputStream 中的类使用一致。
【案例】从文件中读取内容
public static void main(String[] args) throws IOException {
String fileName="D:"+File.separator+"hello.txt";
File f=new File(fileName);
char[] ch=new char[100];
Reader read=new FileReader(f);
int count=read.read(ch);
read.close();
System.out.println("读入的长度为:"+count);
System.out.println("内容为"+new String(ch,0,count));
}
字符输出流Writer
Writer 是所有的输出字符流的父类,它是一个抽象类。
CharArrayWriter、StringWriter 是两种基本的介质流,它们分别向Char 数组、String 中写入数据。
PipedWriter 是向与其它线程共用的管道中写入数据,
BufferedWriter 是一个装饰器为Writer 提供缓冲功能。
PrintWriter 和PrintStream 极其类似,功能和使用也非常相似。
OutputStreamWriter 是OutputStream 到Writer 转换的桥梁,它的子类FileWriter 其实就是一个实现此功能的具体类(具体可以研究一SourceCode)。功能和使用和OutputStream 极其类似,后面会有它们的对应图。
【案例】向文件中写入数据
public static void main(String[] args) throws IOException {
String fileName="D:"+File.separator+"hello.txt";
File f=new File(fileName);
Writer out =new FileWriter(f);
// Writer out =new FileWriter(f,true); // true表示追加内容
String str="hello";
out.write(str);
out.close();
}
文件复制
通过FileInputStream和FileOutputStream
public static void main(String[] args) throws IOException {
File inFile=new File("G:"+ File.separator+"input.txt");
File outFile=new File("G:"+ File.separator+"output.txt");
InputStream in=new FileInputStream(inFile);
OutputStream out =new FileOutputStream(outFile);
byte[] ch=new byte[1024];
int length;
while((length = in.read(ch)) > 0){
out.write(ch, 0, length);
}
in.close();
out.flush();
out.close();
}
通过FileChannel
public static void main(String[] args) throws IOException {
File inFile=new File("G:"+ File.separator+"input.txt");
File outFile=new File("G:"+ File.separator+"output.txt");
FileInputStream fi=new FileInputStream(inFile);
FileOutputStream fo =new FileOutputStream(outFile);
FileChannel in = fi.getChannel();//得到对应的文件通道
FileChannel out = fo.getChannel();//得到对应的文件通道
in.transferTo(0, in.size(), out);//连接两个通道,并且从in通道读取,然后写入out通道
in.close();
out.close();
fi.close();
fo.close();
}
FileChannel是java.nio类库提供的。利用现代操作系统底层机制,避免不必要拷贝和上下文切换。
零拷贝技术
流复制的方式:
FileChannel通道的方式
零拷贝技术是指不用将内容拷贝到用户内存空间,也即内存拷贝次数为零。性能会更好。