Java中的转换流、压缩流、序列化流、打印流及应用场景
作者:微扬嘴角
一、转换流
转换流属于字符流,也是一种高级流,用来包装Reader和Writer。转化流是字符流和字节流之间的桥梁。转换输入流为InputStreamReader,把把字节流转化为字符流;转化输出流为OutputStreamWriter,把字符流转化为字节流。
应用场景:字节流想要使用字符流中的方法则使用转换流。
例如,使用转换输入流读取文件:
一个ANSI编码的文件:
public class MyApp { public static void main(String[] args) throws IOException { //1.创建对象并指定字符编码读取 InputStreamReader isr = new InputStreamReader(new FileInputStream("javase-learning\\src\\example\\org\\domain\\c(ANSI).txt"),"GBK"); //2.读取数据 int ch; while((ch = isr.read())!= -1){ System.out.println((char)ch); } //3.释放资源 isr.close(); } }
上述这种写法JDK11有了替代方案,使用FileReader,其实FileReader是InputStreamReader的子类。。
public class MyApp { public static void main(String[] args) throws IOException { //1.创建对象并指定字符编码读取 FileReader fr = new FileReader("javase-learning\\src\\example\\org\\domain\\c(ANSI).txt", Charset.forName("GBK")); //2.读取数据 int ch; while((ch = fr.read())!= -1){ System.out.println((char)ch); } //3.释放资源 fr.close(); } }
再例如,使用转化输出流写文件:
public class MyApp { public static void main(String[] args) throws IOException { //1.创建对象并指定字符编码 OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("javase-learning\\src\\example\\org\\domain\\d.txt"), "GBK"); //2.写出数据 osw.write("你好你好"); //3.释放资源 osw.close(); } }
JDK11替代方案:
public class MyApp { public static void main(String[] args) throws IOException { //1.创建对象并指定字符编码 FileWriter fw = new FileWriter("javase-learning\\src\\example\\org\\domain\\d.txt", Charset.forName("GBK")); //2.写出数据 fw.write("你好你好"); //3.释放资源 fw.close(); } }
再例如,将本地文件的GBK文件转为UTF-8文件:
注意:上述代码运行后d.txt的内容与b.txt的内容用记事本打开结果相同,且d.txt的编码方式为UTF-8。
二、序列化流和反序列化流
序列化流(ObjectOutputStream,也称对象操作输出流)与反序列化流(ObjectInputStream,也称对象操作输入流)属于字节流,也是高级流,是对基本流的包装。
序列化流可以把java中的对象输出到本地文件中。
说明:存储java对象可以用基本流直接存储属性值,还能直接对文件进行修改。而序列化流存储java对象打开文件是看不懂的,而无法直接修改,这样可以防止暂时存储在文件中的数据被用户修改。
(一)序列化流
构造方法:
成员方法:
例如,新建一个User类:
注意:直接使用对象输出流将对象保存到文件中会出现NotSerializableException异常,解决方法是让JavaBean类实现Serializable接口。
注意:可以看出实现了Serializable接口不用重写方法,因为Serializable接口中没有抽象方法,是标记型接口,一旦实现了这个接口,就表示当前的类可以被序列化。
编写代码实现序列化:
结果如下:
(二)反序列化流
注意:
如果将javabean对象通过序列化存入本地文件,如果这时修改javabean类,再去读取原来存储的javabean对象赋给修改后的类创建对象会报错。
为了防止报错,可以在类中定义版本号,这个版本号会存入通过序列化流存入的本地文件。
如果不想把某些属性字段保存到本地文件中,可以在该属性字段前加瞬态关键字transient。
再例如,运用序列化流和反序列化流读写多个对象。
方式一(读之前已知写入了多少个对象):
方式二(读之前不知道写入了多少个对象):
补充:
三、打印流
java.io.PrintStream:打印流
PrintStream 为其他输出流添加了功能,使它们能够方便地打印各种数据值表示形式。
PrintStream特点:
1.只负责数据的输出,不负责数据的读取
2.与其他输出流不同,PrintStream 永远不会抛出 IOException
3.有特有的方法,print,println
void print(任意类型的值)
void println(任意类型的值并换行)
构造方法:
PrintStream(File file):输出的目的地是一个文件
PrintStream(OutputStream out):输出的目的地是一个字节输出流
PrintStream(String fileName) :输出的目的地是一个文件路径
PrintStream extends OutputStream
继承自父类的成员方法:
- public void close() :关闭此输出流并释放与此流相关联的任何系统资源。
- public void flush() :刷新此输出流并强制任何缓冲的输出字节被写出。
- public void write(byte[] b):将 b.length字节从指定的字节数组写入此输出流。
- public void write(byte[] b, int off, int len) :从指定的字节数组写入 len字节,从偏移量 off开始输出到此输出流。
- public abstract void write(int b) :将指定的字节输出流。
注意:
如果使用继承自父类的write方法写数据,那么查看数据的时候会查询编码表 97->a
如果使用自己特有的方法print/println方法写数据,写的数据原样输出 97->97
可以改变输出语句的目的地(打印流的流向)
输出语句,默认在控制台输出
使用System.setOut方法改变输出语句的目的地改为参数中传递的打印流的目的地
static void setOut(PrintStream out)
重新分配“标准”输出流。
到此这篇关于Java中的转换流、压缩流、序列化流、打印流及应用场景的文章就介绍到这了,更多相关java转换流、压缩流、序列化流、打印流内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!