==================================================================
IO体系特别大,猫叔后续还会继续完善更新的!!!
==================================================================
Java中I/O操作主要是指使用 java.io 包下的内容,进行输入、输出操作。
输入(input)也叫做读取数据,输出Output也叫做作写出数据。
IO流操作的是文件。可以往文件写数据,也可以读取文件的数据。
根据数据的流向分为:输入流和输出流。
输入流 :把数据从其他设备上读取到内存中的流。
输出流 :把数据从内存中写出到 其他设备上的流。
通俗理解就是:
以我们写的Java程序为参考:
将文件数据读取到Java中,是输入流;input;
Java程序有数据写到文件中,是输出流:output;
根据数据的类型分为:字节流和字符流。(计算机中的所有数据都是二进制进行存储 0 1)
字节流 :以字节为单位,读写数据的流。 操作的单位是字节,可以操作所有的数据 。一切皆字节。
字符流 :以字符为单位,读写数据的流。专门操作字符文字。
InputStream 这个抽象类是表示字节流输入的所有类的超类。
OutputStream 这个抽象类是表示字节输出流的所有类的超类。
Reader 用于字符输入流的抽象类。
Writer 用于字符输出流的抽象类。
特点:
1.字节流以Stream结尾;
2.这4个顶级父类都是抽象的;
一切文件数据(文本、图片、视频等)在存储时,都是以二进制数字的形式保存,都一个一个的字节,那么传输时一样如此。所以,字节流可以传输任意文件数据。在操作流的时候,我们要时刻明确,无论使用什么样的流对象,底层传输的始终为二进制数据。
java.io.OutputStream 抽象类是表示字节输出流的所有类的超类,将指定的字节信息写出到目的地。
抽象类,使用其子类。
java.io.FileOutputStream 类是文件输出流,用于将数据写出到文件。
第一种:当你创建一个流对象时,必须传入一个文件路径。该路径下,如果没有这个文件,会创建该文件。如果有这个文件,会清空这个文件的数据。(会清空)
第二种:创建对象,参数中都需要传入一个boolean类型的值, true 表示追加数据, false 表示清空原有数据。这样创建的输出流对象,就可以指定是否追加续写了。(实现数据追加续写)
1、使用FileOutputStream类输出的是数字97、98,但是文件显示的是字符;这是因为文本编辑器,自己帮我们转换的;
2、我们如果要换行 \r\n 或者其它操作要注意,操作的是byte类型数据,需要转换;
java.io.InputStream 抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存中。
抽象类,使用其子类。
java.io.FileInputStream 类是文件输入流,从文件中读取字节。
int read;
while((read = fis.read()) != -1){
System.out.print((char)read);
}
如果是使用的数组读取,注意要读取的是数组中的有效字节数。String类和byte数组之间转换使用。
当使用字节流读取文本文件时,可能会有一个小问题。就是遇到中文字符时,可能不会显示完整的字符,那是因为一个中文字符可能占用多个字节存储。所以Java提供一些字符流类,以字符为单位读写数据,专门用于处理文本文件。
java.io.Reader 抽象类是表示用于读取字符流的所有类的超类,可以读取字符信息到内存中。
抽象类,使用其子类。
java.io.FileReader 类是读取字符文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。
Windows系统的中文编码默认是GBK编码表。
字符不是字节,不能之间存储到文件中,文件只能存储字节,中间会有个缓冲区,会先保存到缓冲区中,不会直接进入文件中,使用flush刷新或close关闭的时候,会找到码表转成字节存入到文件中。
(java.io.Reader-java.io.InputStreamReader-java.io.FileReader )
使用while循环,停止条件是没有数据返回值为-1
如果是使用的数组读取,注意要读取的是数组中的有效字符数。String类和char数组之间转换使用。
java.io.Writer 抽象类是表示用于写出字符流的所有类的超类,将指定的字符信息写出到目的地。
抽象类,使用其子类。
java.io.FileWriter 类是写出字符到文件的便利类。构造时使用系统默认的字符编码和默认字节缓冲区。
(java.io.Writer–java.io.OutputStreamWriter–java.io.FileWriter )
未调用flush或close方法,数据只是保存到了缓冲区,并未写出到文件中。
即便是flflush方法写出了数据,操作的最后还是要调用close方法,释放系统资源。
字符流,只能操作文本文件,不能操作图片,视频等非文本文件。
当我们单纯读或者写文本文件时 使用字符流 其他情况使用字节流。
字节流写数据,不关流,文件中有数据;
字符流写数据,不关流文件中没有数据;
flush:刷新流,数据才会写到文件中;flush后还可以继续操作;
close:关闭流,在关闭前会自动flush一次;close后不能再操作;
JDK7以前的处理IO流异常:有点麻烦,操作中要考虑变量声明周期;try{ 可能出现异常代码}catchcatch(异常类名 变量名) { 处理异常代码}finally{关流}
JDK7的处理IO流异常:会自动关流;try (创建流的代码) { 可能出现异常代码} catch(异常类名 变量名) { 处理异常代码}
java.util.Properties 继承于 Hashtable ,来表示一个持久的属性集。
它使用键值结构存储数据,是一个双列集合,每个键及其对应值都是一个字符串。
结合IO流可以方便的将数据写入流中,或从流中读取数据。
(java.util.Map
文件保存的后缀使用.Properties
文件#符号表示注释;中文是Unicode编码;
显示注释内容,是输入的时间;
一行是一个键值对,键和值中间是等号=连接;
缓冲流:能够高效读写,是对基本流对象的一种增强。
缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率。
按照数据类型分类:
字节缓冲流: BufferedInputStream , BufferedOutputStream
字符缓冲流: BufferedReader , BufferedWriter
特点:都是Buffered开头;结尾是父类名;内部自带数组;
创建一个新的缓冲流,真正操作文件依赖参数指定的流;
缓冲流只需要关闭外面的缓冲流,参数传入的流他会自动帮我们关闭
缓冲流快,建议使用。
public void newLine() : 写一个换行符。
public String readLine() : 读一行文字。没有读取到数据返回null;
转换流是字节与字符间的桥梁!读取字节,解码为字符;写出字符,编码为字节。
java.io.InputStreamReader ,是Reader的子类,是从字节流到字符流的桥梁。
它读取字节,并使用指定的字符集将其解码为字符。它的字符集可以由名称指定,也可以接受平台的默认字符集。
java.io.OutputStreamWriter ,是Writer的子类,是从字符流到字节流的桥梁。使用指定的字符集将字符编码为字节。它的字符集可以由名称指定,也可以接受平台的默认字符集。
java.io.ObjectOutputStream 类,将Java对象的原始数据类型写出到文件,实现对象的持久存储。
public final void writeObject (Object obj) : 将指定的对象写出。
1、该类必须实现 java.io.Serializable 接口, Serializable 是一个标记接口(没有任何内容,起到标识作用),不实现此接口的类将不会使任何状态序列化或反序列化,会抛出 NotSerializableException 。
2、该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用transient 关键字修饰。
ObjectInputStream反序列化流,将之前使用ObjectOutputStream序列化的原始数据恢复为对象。
public final Object readObject () : 读取一个对象。
1、注意从文件中读取对象返回的是Object,取出来做个类型转换。
2、对于JVM可以反序列化对象,它必须是能够找到class文件的类。如果找不到该类的class文件,则抛出一ClassNotFoundException异常。
3、当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个 InvalidClassException 异常。
3.1修改类导致serialVersionUID发生改变–版本号问题;
Serializable 接口给需要序列化的类,提供了一个序列版本号。
该版本号的目的在于验证序列化的对象和对应类是否版本匹配,可以给一个固定的版本号。
3.2该类包含未知数据类型、该类没有可访问的无参数构造方法等修改错误;
打印流分类:
字节打印流:PrintStream;
字符打印流:PrintWrite;
打印流的特点:都是输出的流;原样输出;
平时调用 print 方法和 println 方法,这两个方法都来自于java.io.PrintStream 类,该类能够方便地打印各种数据类型的值,是一种便捷的输出方式。
(java.io.OutputStream–java.io.FilterOutputStream–java.io.PrintStream )
System.out 就是 PrintStream 类型的,只不过它的流向是系统规定的,打印在控制台上。
Stream;
字符打印流:PrintWrite;
打印流的特点:都是输出的流;原样输出;
平时调用 print 方法和 println 方法,这两个方法都来自于java.io.PrintStream 类,该类能够方便地打印各种数据类型的值,是一种便捷的输出方式。
(java.io.OutputStream–java.io.FilterOutputStream–java.io.PrintStream )
System.out 就是 PrintStream 类型的,只不过它的流向是系统规定的,打印在控制台上。
System.setOut(PrintStream out)-------设置out之后,后面的sout打印的内容就会到PrintStream流里面去;可以修改out,打印到文件中。