IO流:I : Input:输入:读
O : Output:输出:写
通过IO可以完成硬盘文件的读和写。
分类方式:
1.以内存作为参照物:
1.输入流:磁盘——>内存:往内存中去,叫做输入(Input)。或者叫做读(Read)。
2.输出流:内存——>磁盘:从内存中出来,叫做输出(Output)。或者叫做写(Write)。
2.读取数据方式不同:
1.字节流:按照字节的方式读取数据,一次读取1个字节bytee,等同于一次读取8个二进制位。
这种流是万能的,什么类型的文件都可以读取。包括:文本文件,图片,声音文件,视频文件等…
假设文件file1.txt,采用字节流的话是这样读的:
a中国bc张三fe
第一次读:一个字节,正好读到’a’
第二次读:一个字节,正好读到’中’字符的一半。
第三次读:一个字节,正好读到’中’字符的另外一半。
2.字符流:按照字符的方式读取数据,一次读取一个字符,这种流不能读取:图片、声音、视频等文件。只能读取纯
文本文件(能以记事本方式打开编辑的),连word文件都无法读取。
假设文件file1.txt,采用字符流的话是这样读的:
a中国bc张三fe
第一次读:'a’字符('a’字符在windows系统中占用1个字节。)
第二次读:'中’字符('中’字符在windows系统中占用2个字节。)
总结:流的分类:
1.输入流 2.输出流 3.字节流 4.字符流
1.java.io.InputStream 字节输入流(读)(万能)
2.java.io.OutputStream 字节输出流(写)(万能)
3.java.io.Reader 字符输入流(读)(普通文本)
4.java.io.Writer 字符输出流(写)(普通文本)
它们都是抽象类。(abstract class)
所有的流都实现了:
java.io.Closeable接口,都是可关闭的,都有close()方法。
流是一个管道,是内存和硬盘之间的通道,用完之后一定要关闭,不然会耗费(占用)很多资源。
所有的输出流都实现了:
java.io.Flushable接口,都是可刷新的,都有flush()方法。
输出流在最终输出之后,一定要记得flush()刷新一下。这个刷新表示将通道/管道当中剩余未输出的数据强行输出完(清空管道!)
刷新的作用就是清空管道。
注意:如果没有flush()可能会导致丢失数据。
重点:在java中只要“类名”以Stream结尾的都是字节流。以“Reader/Writer”结尾的都是字符流。
文件专属:
java.io.FileInputStream(掌握)
java.io.FileOutputStream(掌握)
java.io.FileReader
java.io.FileWriter
转换流:(将字节流转换成字符流)
java.io.InputStreamReader
java.io.OutputStreamWriter
缓冲流专属:
java.io.BufferedReader
java.io.BufferedWriter
java.io.BufferedInputStream
java.io.BufferedOutputStream
数据流专属:
java.io.DataInputStream
java.io.DataOutputStream
标准输出流:
java.io.PrintWriter
java.io.PrintStream(掌握)
对象专属流:
java.io.ObjectInputStream(掌握)
java.io.ObjectOutputStream(掌握)
文件字节输入流:万能:字节方式,输入操作:读:硬盘——> 内存。
方法:
1.read()方法:从输入流读取数据的下一个字节,返回int,如果没有数据了,返回-1。
(效率太慢,可以用while循环,也还是效率低,不建议用)。
2.read(byte[] b):就是读了几个字节就返回这个数字,不是返回字节。读了4个字节就返回4这个数字,而不是这4个字节是什么。
3.close():关闭资源。
利用byte数组读:
规定byte数组读的长度,一次读规定的长度,用read(byte[] b)输出读了多少字节,将数组转换成字符串输出。
数据读完了再读就返回-1。
文件字节输出流:万能:字节方式,输出操作:写:内存——> 磁盘。
1.write(byte[] b):将b个字节的数组写入输入流。(将字符串转化成数组写入输出)
2.flush():彻底完成输入清空缓存器。
3.close():关闭。
文件字符输入流:普通文本:字符方式,输入操作:读:硬盘——> 内存。
FileWriter:
文件字符输出流:普通文本:字符方式,输出操作:写:内存——> 硬盘。
InputStreamReader:
文件字符输入流:普通文本:字符方式,输入操作:读:硬盘——> 内存。
将字节流转换成字符流:
InputStreamReader(FileInputStream in):里面传一个FileInputStream类型对象进行转换。
OutputStreamWriter:同理:将字节流转换成字符流:
OutputStreamWriter(FileOutputStream in):里面传一个FileOutputStream类型对象进行转换。
BufferedReader:
文件字符输入流:普通文本:字符方式,输入操作:读:硬盘——> 内存。
带有缓冲区的字符输入流。
使用这个流的时候不需要自定义char数组,或者说不需要自定义byte数组。自带缓冲。
方法:
readLine():读取一行,没有数据输出null
BufferedReader(InputStreamReader in)
BufferedReader(FileReader fr)
// 当一个流的构造方法中需要一个流的时候,这个被传进来的流叫做:节点流。
// 外部负责包装的这个流,叫做:包装流,还有一个名字叫做:处理流。
// 像当前这个程序来说:FileReader就是一个节点流。BufferedReader就是包装流/处理流。
// 对于包装流来说,只需要关闭最外层流就行,里面的节点流会自动关闭。(可以看源代码。)
BufferedWriter同理:
文件字符输出流:普通文本:字符方式,输出操作:写:内存——> 硬盘。
DataInputStream:
文件字节输入流:万能:字节方式,输入操作:读:硬盘——> 内存。
DataOutputStream写的文件,只能使用DataInputStream去读。并且读的时候你需要提前知道写入的顺序。
读的顺序需要和写的顺序一致。才可以正常取出数据。
DataOutputStream:
数据字节输出流:万能:字节方式,输出操作:写:内存——> 磁盘。
数据专属的流。这个流可以将数据连同数据的类型一并写入文件。注意:这个文件不是普通文本文档。(这个文件使用记事本打不开。)
参与序列化和反序列化的对象,必须实现Serializable接口。
通过源代码发现,Serializable接口只是一个标志接口
public interface Serializable {
}
起到标识的作用,标志的作用,java虚拟机看到这个类实现了这个接口,可能会对这个类进行特殊待遇。
Serializable这个标志接口是给java虚拟机参考的,java虚拟机看到这个接口之后,会为该类自动生成
一个序列化版本号。
transient关键字表示游离的,不参与序列化。
private transient String name; // name不参与序列化操作!
//java语言首先通过类名进行比对,如果类名不一样,肯定不是同一个类。
//如果类名一样,再怎么进行类的区别?靠序列化版本号进行区分。
//IDEA工具自动生成序列化版本号。
//private static final long serialVersionUID = -7998917368642754840L;
//Java虚拟机看到Serializable接口之后,会自动生成一个序列化版本号。
//这里没有手动写出来,java虚拟机会默认提供这个序列化版本号。
//这种自动生成的序列化版本号缺点是:一旦代码确定之后,不能进行后续的修改,
// 建议将序列化版本号手动的写出来。不建议自动生成
//凡是一个类实现了Serializable接口,建议给该类提供一个固定不变的序列化版本号。
标准的字节输出流。默认输出到控制台。