1. 内存流
1.1 什么是内存流
当输出流的目的,和输入流的源是内存时,这样的流称之为内存流。
ByteArrayOutputStream:内存流的输出流
ByteArrayInputStream:内存流的输入流,它是唯一一种能够直接从网络上获取二进制数据的流
CharArrayReader:内存流中的输入流
CharArrayWriter:内存流中的输出流
ByteArrayInputStream主要完成将内容从内存读入程序之中,而ByteArrayOutputStream的功能主要是将数据写入到内存中。
1.2 内存流构造方法
ByteArrayInputStream是输入流的一种实现,它有两个构造函数,每个构造函数都需要一个字节数组来作为其数据源:
ByteArrayInputStream(byte[] buf)
ByteArrayInputStream(byte[] buf,int offse , int length)
ByteArrayOutputStream是输出流的一种实现,它有两个构造函数:
ByteArrayOutputStream():内存中分配了一个字节数组。
BuyteArrayoutputStream(int):创建一个新的 byte 数组输出流,它具有指定大小的缓冲区容量(以字节为单位)
1.3 写数据到内存流和从内存流读数据
如果程序在运行过程中要产生一些临时文件,可以采用虚拟文件方式实现(其实是一段内存),JDK中提供了ByteArrayInputStream和ByteArrayOutputStream两个类可实现类似于内存虚拟文件的功能。
通过内存流写数据:
package cn.sz.gl.test05;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
public class Test {
public static void writeByteArray(String msg) {
ByteArrayOutputStream baos = null;
try {
baos = new ByteArrayOutputStream();
baos.write(msg.getBytes());
baos.flush();
// 1.通过toString()得到缓冲区数据
// System.out.println(baos.toString());
// 2.通过toByteArray()方法得到一个byte数组
byte[] b = baos.toByteArray();
String str = new String(b);
System.out.println("str:" + str);
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (baos != null) {
baos.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
String msg = "hello world";
Test.writeByteArray(msg);
}
}
通过内存流读取数据:
package cn.sz.gl.test05;
import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.IOException;
public class Test {
public static void readByteArray(String msg) {
ByteArrayOutputStream baos = null;
ByteArrayInputStream bais = null;
try {
baos = new ByteArrayOutputStream();
baos.write(msg.getBytes());
baos.flush();
bais = new ByteArrayInputStream(baos.toByteArray());
byte [] b = new byte[1024];
int len = bais.read(b);
System.out.println("len:"+len);
String str = new String(b,0,len);
System.out.println("str:"+str);
} catch (IOException e) {
e.printStackTrace();
} finally {
try {
if (baos != null) {
baos.close();
}
} catch (IOException e) {
e.printStackTrace();
}
}
}
public static void main(String[] args) {
String msg = "hello world";
readByteArray(msg);
}
}
2.缓冲流(处理流)(重点)
缓冲流是建立在相应的节点流之上,对读写的数据提供了缓冲的功能,提高了读写的效率,还增加了一些新的方法。
JDK提供四种缓冲流:
BufferedInputStream 可以对任何的InputStream流进行包装
BufferedOutputStream 可以对任何的OutputStream流进行包装
BufferedReader 可以对任何的Reader流进行包装
BufferedWriter 可以对任何的Writer流进行包装
注意:
对于缓冲输出流,写出的数据会先缓存在内存缓冲区中,关闭此流前要用flush()方法将缓存区的数据立刻写出。
关闭过滤流时,会自动关闭缓冲流包装的所有底层流。
2.1 字节缓冲流
字节缓冲分输入缓冲和输出缓冲。
1、 缓冲字节输入流:BufferedInputStream。BufferedInputStream为另一个输入流添加一些功能,即缓冲输入以及支持mark
和reset
方法的能力。
2、 缓冲字节输出流:BufferedOutputStream。 输出字节时,先把要输出的字节输出到缓冲区,当手动调用flush()方法或者缓冲区满或者流关闭时才会把数据输出到结点流。
3、字节缓冲输入流的缓冲区默认大写为8k字节。也可以指定缓冲区大小
2.2 字符缓冲流
字符缓冲分输入缓冲和输出缓冲。
1、 BufferedReader 从字符输入流中读取文本,缓冲各个字符,从而实现字符、数组和行的高效读取。 mark和 reset方法。新增了readLine()方法用于一次读取一行字符串(以‘\r’或‘\n’认为一行结束)。
2、 BufferedWriter 将文本写入字符输出流,缓冲各个字符,从而提供单个字符、数组和字符串的高效写入。
3、字符缓冲输入流的缓冲区默认大写为8k个字符。也可以指定缓冲区大小。
3. 转换流(重点)
前面已经讲过,Java支持字节流和字符流,但有时需要字节流和字符流之间的转换。
InputStreamReader 和OutputStreamWriter,这两个类是将字节流转换为字符流的类,InputStreamReader 可以将一个InputStream转换为Reader,OutputStreamWriter可以将一个OutputStream转换为Writer。
InputStreamReader有两个主要的构造函数:
InputStreamReader(InputStream in)
// 用默认字符集创建一个InputStreamReader对象
InputStreamReader(InputStream in,String CharsetName)
// 接受已指定字符集名的字符串,并用该字符集创建对象
OutputStreamWriter也有对应的两个主要的构造函数:
OutputStreamWriter(OutputStream in)
// 用默认字符集创建一个OutputStreamWriter对象
OutputStreamWriter(OutputStream in,String CharsetNarme)
// 接受已指定字符集名的字符串,并用该字符集创建OutputStreamWriter对象
为了达到最高的效率,避免频繁地进行字符与字节间的相互转换,最好不要直接使用这两个类来进行读写,应尽量使用BufferedWriter类包装OutputStreamWriter类,用BufferedReader类包装InputStreamReader类。例如:
BufferedWriterout = new BufferedWriter(new OutputStreamWriter(System.out));
BufferedReaderin = new BufferedReader(new InputStreamReader(System.in));
接着从一个实际的应用中来了解InputStreamReader的作用,用一种简单的方式读取键盘上输入的一行字符:
BufferedReader in = newBufferedReader(new InputStreamReader(System.in));
String strLine =in.readLine();
可见,构建BufferedReader对象时,必须传递一个Reader类型的对象作为参数,而键盘对应的System.in是一个InputStream 类型的对象,所以这里需要用到一个InputStreamReader的转换类,将System.in转换成字符流之后,放入到字符流缓冲区之中,之后从缓冲区中每次读入一行数据
例如:
package cn.sz.gl.test05;
import java.io.BufferedReader;
import java.io.IOException;
import java.io.InputStreamReader;
public class Test {
public static void main(String args[]) {
BufferedReader buf = new BufferedReader(new InputStreamReader(System.in));
String str = null;
while (true) {
System.out.print("请输入数字:");
try {
str = buf.readLine();
} catch (IOException e) {
e.printStackTrace();
}
int i = -1;
try {
i = Integer.parseInt(str);
i++;
System.out.println("输入的数字修改后为:" + i);
break;
} catch (Exception e) {
System.out.println("输入的内容不正确,请重新输入!");
}
}
}
}
4. 打印流
4.1 打印流构造方法
在整个IO包中,打印流是输出信息最方便的类,主要包括字节打印流(PrintStream)和字符打印流(PrintWriter)。打印流提供了非常方便的打印功能,可以打印任何的数据类型。如:小数、整数、字符串等。
PrintStream和PrintWriter都属于输出流,分别针对输出字节和字符。
PrintStream和PrintWriter提供了重载的print()、println()方法用于多种数据类型的输出。
PrintStream和PrintWriter不会抛出异常,用户通过检测错误状态获取错误信息。
PrintStream和PrintWriter有自动flush 功能。
·PrintStream类有下面几个构造方法:
PrintStream(OutputStream out)
PrintStream(OutputStream out, boolean auotflush)
PrintStream(OutputStream out, boolean auotflush, String encoding)
·PrintWriter类有下面几个构造方法:
PrintWriter(OutputStream out)
PrintWriter(OutputStream out, boolean autoflush)
PrintWriter(Writer out)
PrintWriter(Writerout, boolean autoflush)
其中autoflush控制在Java中遇到换行符(\n)时是否自动清空缓冲区,encoding是指定编码方式。
4.2 打印流常用方法
PrintWriter即使遇到换行符(\n)也不会自动清空缓冲区,只在设置了autoflush模式下使用了println方法后才自动清空缓冲区。PrintWriter相对PrintStream最有利的一个地方就是println方法的行为,在Windows的文本换行是"\r\n",而Linux下的文本换行是"\n",如果希望程序能够生成平台相关的文本换行,而不是在各种平台下都用"\n"作为文本换行,那么就应该使用PrintWriter的println方法时,PrintWriter的println方法能根据不同的操作系统而生成相应的换行符。
控制台打印:
package cn.sz.gl.test05;
import java.io.PrintWriter;
public class Test {
public static void main(String args[]){
// 通过System.out为PrintWriter实例化
PrintWriter out = new PrintWriter(System.out);
// 向屏幕上输出
out. println("Hello World!");
out.close(); //如果此句不写,则没有内容,跟PrintStream有区别
}
}
在文件中打印:
package cn.sz.gl.test05;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.io.PrintWriter;
public class Test {
public static void main(String args[]) {
PrintWriter out = null;
File f = new File("c:\\temp.txt");
try {
// 由FileWriter实例化,则向文件中输出
out = new PrintWriter(new FileWriter(f));
} catch (IOException e) {
e.printStackTrace();
}
out.print("Hello World!" + "\r\n");
out.close();
}
}
5. 对象流
5.1 对象的序列化
Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。通过将对象序列化,可以方便的实现对象的传输及保存。
Java序列化是指把Java对象转换为字节序列的过程;而Java反序列化是指把字节序列恢复为Java对象的过程。当两个进程进行远程通信时,可以相互发送各种类型的数据,包括文本、图片、音频、视频等,而这些数据都会以二进制序列的形式在网络上传送。那么当两个Java进程进行通信时,实现进程间的对象传送,就需要Java序列化与反序列化了。换句话说,一方面,发送方需要把这个Java对象转换为字节序列,然后在网络上传送;另一方面,接收方需要从字节序列中恢复出Java对象。
5.2 对象输入流和输出流
在Java中提供了ObjectlnputStream与ObjectOutputStream这两个类用于序列化对象的操作。使用对象输出流输出序列化对象的步骤,有时也称为序列化。使用对象输入流读入对象的过程,有时也称为反序列化。
这两个类是用于存储和读取对象的输入输出流类,不难想象,只要把对象中的所有成员变量都存储起来,就等于保存了这个对象,之后从保存的对象之中再将对象读取进来就可以继续使用此对象。ObjectInputStream与ObjectOutputStream类,可以帮开发者完成保存和读取对象成员变量取值的过程,但要求读写或存储的对象必须实现了java.io.Serializable接口,但Serializable接口中没有定义任何方法,仅仅被用作一种标记,以被编译器作特殊处理。
如下范例所示:
import java.io.*;
public class Person implements Serializable{
private String name;
private int age;
public Person(String name,int age){
this.name = name;
this.age = age;
}
public String toString(){
return " 姓名:"+this.name+",年龄:"+this.age;
}
}
第2行所中,类Person实现了Serializable接口,所以此类的对象可序列化。下面的范例使用ObjectOutputStream与ObjectInputStream将Person类的对象保存在文件之中
package cn.sz.gl.test05;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.InputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.OutputStream;
public class Test {
// 以下方法为序列化对象方法,将对象保存在文件之中
public static void serialize(File f) throws Exception{
OutputStream outputFile = new FileOutputStream(f);
ObjectOutputStream cout = new ObjectOutputStream(outputFile)
cout.writeObject(new Person("张三",25));
cout.close();
}
// 以下方法为反序列化对象方法,从文件中读取已经保存的对象
public static void deserialize(File f) throws Exception {
InputStream inputFile = new FileInputStream(f);
ObjectInputStream cin = new ObjectInputStream(inputFile);
Person p = (Person) cin.readObject();
System.out.println(p);
}
public static void main(String args[]) {
File f = new File("SerializedPerson");
serialize(f);
deserialize(f);
}
}
5.2.1 serialVersionUID 常量
在对象进行序列化或反序列化操作的时候,要考虑 JDK 版本的问题。如果序列化的 JDK 版本和反序列化的 JDK 版本不统一,则可能造成异常。因此在序列化操作中引入了一个serialVersionUID 的常量来验证版本的一致性。在进行反序列化时,JVM 会把传来的字节流中的 serialVersionUID 与本地相应实体(类)的serialVersionUID 进行比较。如果相同就认为是一致的,可以进行反序列化,否则就会出现反序列化版本不一致的异常
Idea 配置自动生成序列号:
在需要生成序列化id类上: alt + enter
5.2.2 transient关键字
如果不希望类中的属性被序列化,可以在声明属性之前加上transient关键字。如下所示,下面的代码修改自前面所用到的Person.java程序,在声明属性时,前面多加了一个transient关键字
private transient String name;
private transient int age;
注意:
序列化细节:
1) 被序列化的类的内部的所有属性,必须是可序列化的
2) static,transient修饰的属性,不可以被序列化