一、字节缓冲流
1.1 概述
* 缓冲流,也叫高效流,是对4个基本的`FileXxx` 流的增强,所以也是4个流,按照数据类型分类:
- 字节缓冲流:`BufferedInputStream`,`BufferedOutputStream`
- 字符缓冲流:`BufferedReader`,`BufferedWriter`
* 缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率。
1.2 字节缓冲输出流BufferedOutputStream
* java.io.BufferedOutputStream extends OutputStream
BufferedOutputStream:字节缓冲输出流
* 继承自父类的共性成员方法:
- 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) : 将指定的字节输出。
* 构造方法:
BufferedOutputStream(OutputStream out) 创建一个新的缓冲输出流,以将数据写入指定的底层输出流。
BufferedOutputStream(OutputStream out, int size) 创建一个新的缓冲输出流,以将具有指定缓冲区大小的数据写入指定的底层输出流
参数:
OutputStream out:字节输出流
我们可以传递FileOutputStream,缓冲流会给FileOutputStream增加一个缓冲区,提高FIleOutputStream的写入效率
int size : 指定缓冲流内部缓冲区的大小,不指定默认大小
* 使用步骤:
1. 创建一个FileOutputStream对象,构造方法中绑定要输出的目的地
2. 创建BufferOutputStream对象,构造方法中传递FileOutputStream对象,提高FileOutputStream对象效率
3. 使用BufferedOutputStream对象中的方法write,把数据写入到内部缓冲区中
4. 使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
5. 释放资源(会先调用flush方法刷新数据,故第四步可以省略)
import com.sun.xml.internal.ws.policy.privateutil.PolicyUtils;
import java.io.*;
public class Demo01BufferedOutStream {
public static void main(String[] args) throws IOException {
//1. 创建一个FileOutputStream对象,构造方法中绑定要输出的目的地
FileOutputStream fos = new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\a.txt");
//2. 创建BufferOutputStream对象,构造方法中传递FileOutputStream对象,提高FileOutputStream对象效率
BufferedOutputStream bos = new BufferedOutputStream(fos);
//3. 使用BufferedOutputStream对象中的方法write,把数据写入到内部缓冲区中
bos.write("我把数据写入内部缓冲区中".getBytes());
//4. 使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
bos.flush();
//5. 释放资源(会先调用flush方法刷新数据,故第四步可以省略)
bos.close();
}
}
1.3 字节缓冲输入流
java.io.BufferedInputStream extends InputStream
BufferedInputStream:字节缓冲输入流
* 继承自父类的成员方法:
int read() 从输入流中读取数据的下一字节
int read(byte[] b)从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b 中
void close() 关闭此输入流并释放与该流关联的所有系统资源。
* 构造方法:
BufferedInputStream(InputStream in):创建一个 BufferedInputStream 并保存其参数,即输入流in,以便将来使用。
BufferedInputStream(InputStream in , int size):创建具有指定缓冲区大小的 BufferedInputStream 并保存其参数,即输入流 in, 以便将来使用
参数:
InputStream in:字节输入流
我们可以传递FileInputStream,缓冲流会给FileInputStream增加一个缓冲区,提高FileInputStream的读取效率
int size:指定缓冲流内部缓冲区大小,不指定默认
* 使用步骤(重点):
1. 创建FileInputStream对象,构造方法中绑定要读取的数据源
2. 创建BufferedInputStream对象,构造方法中传递FilrInputStream对象,提高FileInputStream的读取效率
3. 使用BufferedInputStream对象中的方法read,读取文件
4. 释放资源
import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.IOException;
public class Demo02BufferedInputStream {
public static void main(String[] args) throws IOException {
//1. 创建FileInputStream对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\a.txt");
//2. 创建BufferedInputStream对象,构造方法中传递FilrInputStream对象,提高FileInputStream的读取效率
BufferedInputStream bis = new BufferedInputStream(fis);
//3. 使用BufferedInputStream对象中的方法read,读取文件
//int read() 从输入流中读取数据的下一字节
/*int len = 0; //记录每次读取到的字节
while((len = bis.read()) != -1){
System.out.println(len);
}*/
//int read(byte[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组b中。
byte[] bytes = new byte[1024];//存储每次读取的数据
int len = 0; //记录每次读取的有效字节个数
while ((len = bis.read(bytes)) != -1) {
System.out.println(new String(bytes, 0, len));
}
//4. 释放资源
bis.close();
}
}
1.4 缓冲流的效率测试
* 文件复制练习:读
* 明确:
数据源:C:\Users\32189\Desktop\考研资料\a\a.jpg
数据的目的地:d:\\b.jpg
* 文件复制的步骤:
1. 创建一个字节输入流对象,构造方法中绑定要读取放入数据源
2. 创建一个字节输出流对象,构造方法中绑定要写如的目的地
3. 使用字节输入流对象中的方法read读取文件
4. 使用字节输出流中的方法write, 把读取到的字节写入到目的地的文件中
5. 释放资源
文件大小:261,958 字节
一次读写一个字节:2636毫秒
优化,使用数组缓冲读取多个字节,写入多个字节:18毫秒
import java.io.FileInputStream;
import java.io.FileOutputStream;
import java.io.IOException;
public class Demo01CopyFile {
public static void main(String[] args) throws IOException {
long s = System.currentTimeMillis();
//1. 创建一个字节输入流对象,构造方法中绑定要读取的数据源
FileInputStream fis = new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\a\\a.jpg");
//2. 创建以一个字节输出流对象,构造方法中要绑定写入的目的地
FileOutputStream fos = new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\b\\b.jpg");
//一次读取一个字节写入一个字节的方式
//3. 使用字节输入流对象中的方法tead读取文件
int len = 0;
while ((len = fis.read()) != -1) {
//4. 使用字节输出流中的方法write, 把读取到的字节写入到目的地的文件中
fos.write(len);
}
/* //优化,使用数组缓冲读取多个字节,写入多个字节
byte[] bytes = new byte[1024];
//3.使用字节输入流对象中的方法read读取文件
int len = 0;
while((len = fis.read(bytes))!=-1){
//4. 使用字节输入流中的方法write,把读取到的字节写入到目的地的文件中
fos.write(bytes,0,len);
}*/
//5. 释放资源(先关闭写的,后关闭读的;如果写完了,肯定读取完毕了)
fos.close();
fis.close();
long e = System.currentTimeMillis();
System.out.println("复制文件共耗时"+(e-s)+"毫秒");
}
}
* 文件复制练习:写
* 明确:
数据源:C:\Users\32189\Desktop\考研资料\a\a.jpg
数据目的地:C:\Users\32189\Desktop\考研资料\b\b.jpg
* 文件复制的步骤:
1. 创建字节缓冲输入流对象,构造方法中传递字节输入流
2. 创建字节缓冲输出流对象,构造方法中传递字节输出流
3. 使用字节缓冲输入流对象中的方法read,读取文件
4. 使用字节缓冲输出流中的方法write,把读取的数据写入到内部缓冲区中
5. 释放资源(会先把缓冲区中的数据,刷新到文件中)
* 文件大小:261,958 字节
一次读写一个字节:37毫秒
优化,使用数组缓冲读取多个字节,写入多个字节:5毫秒
import java.io.*;
public class Demo02CopyFile {
public static void main(String[] args) throws IOException {
long s = System.currentTimeMillis();
//1. 创建字节缓冲输入流对象,构造方法中传递字节输入流
BufferedInputStream bis = new BufferedInputStream(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\a\\a.jpg"));
//2. 创建字节缓冲输出流对象,构造方法中传递字节输出流
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\b\\b.jpg"));
//3. 使用字节缓冲输入流对象中的方法read,读取文件
/*//一次读取一个字节写入一个字节的方式
int len = 0;
while ((len = bis.read()) != -1){
bos.write(len);
}*/
//使用数组缓冲读取多个字节,写入多个字节
byte[] bytes = new byte[1024];
int len = 0;
while((len = bis.read(bytes)) != -1){
bos.write(bytes,0,len);
}
bos.close();
bis.close();
long e = System.currentTimeMillis();
System.out.println("复制文件共耗时" + (e - s) + "毫秒");
}
}
二、字符缓冲流
2.1 字符缓冲输出流 BufferedWriter
java.io.BufferedWriter extends Writer
BufferedWrite:字符缓冲输出流
* 继承自父类的共性成员方法:
- void write(char[] cbuf)写入字符数组。
- abstract void write(char[] chuf, int off, int len)写入字符数组的某一部分,off数组的开始索引,len写的字符个数。
- void write(String str) 写入字符串
- void write(String str, int off, int len) 写入字符串的某一部分,off字符串的开始索引,len写的字符个数。
- void flush() 刷新该流的缓冲。
- void close() 关闭此流,但要先刷新它。
* 构造方法:
BufferedWrite(Writer out) 创建一个使用默认大小输出缓冲区的缓冲字符输出流。
BufferedWrite(Writer out, int sz) 创建一个使用给定大小输出缓冲区的新缓冲字符输出流。
参数:
Writer out:字符输出流
我们可以传递FileWriter,缓冲流会给FileWriter增加一个缓冲区,提高FileWriter的写入效率
int sz: 指定缓冲区的大小,不写默认大小
* 特有的成员方法:
void newLine() : 写入一个行分隔符。会根据不同的操作系统,获取不同的行分隔符
换行:换行符号
windows: \r\n
linux:/n
mac:/r
* 使用步骤:
1. 创建字符缓冲输出流对象,构造方法中传递字符输出流
2. 调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
3. 调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
4. 释放资源
import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
public class Demo03BufferedWriter {
public static void main(String[] args) throws IOException {
//1. 创建字符缓冲输出流对象,构造方法中传递字符输出流
BufferedWriter bw = new BufferedWriter(new FileWriter("C:\\Users\\32189\\Desktop\\考研资料\\c.txt"));
//2. 调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
for (int i = 0; i < 10; i++){
bw.write("传智播客");
//bw.write("\r\n");
bw.newLine(); //换行
}
//3. 调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
bw.flush();
bw.close();
}
}
2.2 字符缓冲输入流 BufferedReader
java.io.BufferedReader extends Reader
* 继承自父类的共性成员方法:
int read() 读取单个字符并返回。
int read(char[] cbuf)一次读取多个字符,将字符读入数组。
void close() 关闭该流并释放与之关联的所有资源。
* 构造方法:
BufferedReader(Reader in) 创建一个使用默认大小输入缓冲区的缓冲字符输入流。
BufferedReader(Reader in, int sz) 创建一个使用指定大小输入缓冲区的缓冲字符输入流。
参数:
Reader in:字符输入流
我们可以传递FileReader,缓冲流会给FileReader增加一个缓冲区,提高FileReader的读取效率
* 特有的成员方法:
String readLine():读一行文本。读取一行数据
注:行终止符号:通过下列字符之一即可认为某行已终止:换行('\n')、回车('\r')或回车后直接跟着换行(\r\n)。
返回值:
包含该行内容的字符串,不包含任何行终止符,如果已经到达流末尾,则返回null
* 使用步骤:
1. 创建字符缓冲输入流对象,构造方法中传递字符输入流
2. 使用字符缓冲输入流对象中的方法read/readLine读取文本
3. 释放资源
import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;
public class Demo04BufferedReader {
public static void main(String[] args) throws IOException {
//1. 创建字符缓冲输入流对象,构造方法中传递字符输入流
BufferedReader br = new BufferedReader(new FileReader("C:\\Users\\32189\\Desktop\\考研资料\\c.txt"));
//2. 使用字符缓冲输入流对象中的方法read/readLine读取文本
/* String line = br.readLine();
System.out.println(line);
line = br.readLine();
System.out.println(line);
line = br.readLine();
System.out.println(line);
line = br.readLine();
System.out.println(line);*/
/*
发现以上读取是一个重复的过程,所以可以使用训话优化
不知道文件中有多少行数据,所以使用while循环
while的结束条件,读取到null结束
*/
String line;
while((line = br.readLine()) != null){
System.out.println(line);
}
//3. 释放资源
br.close();
}
}
2.3 练习:对文本文档中的乱序段落进行升序排序
* 练习:
对文本的内容进行排序
按照(1,2,3,4....)顺序排序
* 分析:
1. 创建一个HashMap集合对象,key:存储每行文本的序号(1,2,3...);value:存储每行的文本
2. 创建字符缓冲输入流对象,构造方法中绑定字符输入流
3. 创建字符缓冲输入流对象,构造方法中绑定字符输入流
4. 使用字符缓冲输入流中的方法readLine,逐行读取文本
5. 对读取到的文本进行切割,获取行中的序号和文本内容
6. 把切割好的序号和文本的内容存储到HashMap集合中(key序号是有序的,会自动排序1,2,3,4..)
7. 遍历HashMap集合,获取每一个键值对
8. 把每一个键值对,拼接为一个文本行
9. 把拼接好的文本,使用字符缓冲输入流中的方法write,写入到文件中
10.释放资源
import java.io.*;
import java.util.HashMap;
import java.util.Map;
public class Demo05Test {
public static void main(String[] args) throws IOException {
//1. 创建一个HashMap集合对象,key:存储每行文本的序号(1,2,3...);value:存储每行的文本
HashMap<String, String> map = new HashMap<>();
//2. 创建字符缓冲输入流对象,构造方法中绑定字符输入流
BufferedReader br = new BufferedReader(new FileReader("C:\\Users\\32189\\Desktop\\考研资料\\in.txt"));
//3. 创建字符缓冲输入流对象,构造方法中绑定字符输入流
BufferedWriter bw = new BufferedWriter(new FileWriter("C:\\Users\\32189\\Desktop\\考研资料\\out.txt"));
//4. 使用字符缓冲输入流中的方法readLine,逐行读取文本
String line;
while((line = br.readLine())!= null){
//5. 对读取到的文本按.进行切割,获取行中的序号和文本内容
String[] arr = line.split("\\.");
//6. 把切割好的序号和文本的内容存储到HashMap集合中(key序号是有序的,会自动排序1,2,3,4..)
map.put(arr[0],arr[1]);
}
//7.遍历HashMap集合,获取每一个键值对
for(String key : map.keySet()){
String value = map.get(key);
//8.把每一行键值对,拼接为一个文本行
line = key + "."+value;
//9. 把拼接好的文本,使用字符缓冲输入流中的方法write,写入到文件中
bw.write(line);
bw.newLine(); //写换行
}
//10.释放资源
bw.close();
br.close();
}
}
三、字符编码和字符集
import java.io.FileReader;
import java.io.IOException;
/*
FileReader可以读取默认编码格式(UTF-8)的文件
FileReader读取系统默认编码(中文GBK)会产生乱码
*/
public class Demo01FileReader {
public static void main(String[] args) throws IOException {
FileReader fr = new FileReader("C:\\Users\\32189\\Desktop\\考研资料\\我是GBK格式的文档.txt");
int len = 0;
while((len = fr.read())!=-1){
System.out.print((char) len);//����GBK��ʽ���ĵ�
}
fr.close();
}
}
3.1 转换流
利用InputStreamReader可以把某字节输入流用特定的编码表读取为字符。
利用OutputStreamWriter可以把某字符用特定的编码表转换为字节,存储到硬盘中。
使用OutputStreamWriter 指定编码表,将字符编码为字节。
* java.io.OutputStreamWriter extends Writer
OutputStreamWriter:是字符流通向字节流的桥梁:可使用指定的 charset 将要写入流中的字符编码转换成字节(编码:把能看懂的变成看不懂的)
* 继承自父类的共性成员方法:
- void write(int c) 写入单个字符。
- void write(char[] cbuf) 写入字符数组。
- abstract void write(char[] cbuf, int off, int len)写入字符数组的某一部分,off数组的开始索引,len写的字符个数。
- void write(String str)写入字符串。
- void flush()刷新该流的缓冲。
- void close() 关闭此流,但要先刷新它。
* 构造方法:
OutputStreamWriter(OutputStream out) 创建使用默认字符编码的 OutputStreamWriter。
OutputStreamWriter(OutputStream out, String charsetName) 创建使用指定字符集的 OutputStreamWriter。
* 参数:
OutputStream out:字节输出流,可以用来转换之后的字节到文件中。
String charsetName:指定的编码表名称,不区分大小写,可以是 utf-8/UTF-8,gbk/GBK,...不指定时默认使用UTF-8
* 使用步骤:
1. 创建一个OutputStreamWriter对象,构造方法中传递字节输出流和指定的编码表名称。
2. 使用OutputStreamWriter对象中的方法writer,把字符转换为字节存储到缓冲区中(编码)
3. 使用OutputStreamWriter对象中的方法flush,把内存缓冲区中的8字节刷新到文件中(使用字节流写字节的过程)。
4. 释放资源
import java.io.*;
public class Demo02OutputStreamWriter { //一段文字,用utf到-8编码写gbk.txt里,用gbk编码写到gbk.txt里
public static void main(String[] args) throws IOException {
//write_utf_8();
write_gbk();
}
/*
使用转换流OutputStreamWriter写gbk格式的文件
*/
private static void write_gbk() throws IOException {
//1. 创建一个OutputStreamWriter对象,构造方法中传递字节输出流和指定的编码表名称。
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\gbk.txt"),"gbk");
//2. 使用OutputStreamWriter对象中的方法writer,把字符转换为字节存储到缓冲区中(编码)
osw.write("你好");
//3. 使用OutputStreamWriter对象中的方法flush,把内存缓冲区中的字节刷新到文件中(使用字节流写字节的过程)。
osw.flush();
//4. 释放资源
osw.close();
}
/*
使用转换流OutputStreamWriter写UTF-8格式的文件
*/
private static void write_utf_8() throws IOException {
//1. 创建一个OutputStreamWriter对象,构造方法中传递字节输出流和指定的编码表名称。
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\utf-8.txt"),"utf-8");
//2. 使用OutputStreamWriter对象中的方法writer,把字符转换为字节存储到缓冲区中(编码)
osw.write("你好");
//3. 使用OutputStreamWriter对象中的方法flush,把内存缓冲区中的字节刷新到文件中(使用字节流写字节的过程)。
osw.flush();
//4. 释放资源
osw.close();
}
}
使用InputStreamReader读取指定编码表格式的文件,将字节解码为字符。
java.io.InputStreamReader extends Reader
* InputStreamReader:是字节流通向字符流的桥梁:它使用指定的charset 读取字节并将其解码为字符。(解码:把看不懂的变成能看懂的)
* 继承字符类的共性成员方法:
int read() 读取单个字符并返回。
int read(char[] cbuf)一次读取多个字符,将字符读入数组。
void close() 关闭该流并释放与之关联的所有资源。
* 构造方法:
InputStreamReader(InputStream in) 创建一个使用默认字符集的 InputStreamReader。
InputStreamReader(InputStream in, String charsetName) 创建使用指定字符集的 InputStreamReader。
参数:
InputStream in:字节输入流,用来读取文件中保存的字节。
String charsetName:指定的编码表名称,不区分大小写,可以是 utf-8/UTF-8,gbk/GBK,...不指定时默认使用UTF-8
* 使用步骤:
1. 创建InputStreamReader对象,构造方法中欧传递字节输入流和指定的编码表名称。
2. 使用InputStreamReader对象中的方法read读取文件
3. 释放资源
* 注意事项:
构造方法中指定的编码表名称要和文件的编码相同,否则会发生乱码。
import jdk.internal.util.xml.impl.Input;
import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.InputStreamReader;
public class Demo03InputStreamReader {
public static void main(String[] args) throws IOException {
//read_utf_8();
read_gbk();
}
/*
使用InputStreamReader读取gbk格式的文件
*/
private static void read_gbk() throws IOException {
//1. 创建InputStreamReader对象,构造方法中传递字节输入流和指定的编码表名称。
InputStreamReader isr = new InputStreamReader(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\gbk.txt"),"gbk");
//2. 使用InputStreamReader对象中的方法read读取文件
int len = 0;
while((len = isr.read()) != -1){
System.out.println((char) len); //你好
}
//3. 释放资源
isr.close();
}
/*
使用InputStreamReader读取UTF-8格式的文件
*/
private static void read_utf_8() throws IOException {
//1. 创建InputStreamReader对象,构造方法中欧传递字节输入流和指定的编码表名称。
InputStreamReader isr = new InputStreamReader(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\utf-8.txt"),"UTF-8");
//2. 使用InputStreamReader对象中的方法read读取文件
int len = 0;
while((len = isr.read()) != -1){
System.out.println((char) len); //你好
}
//3. 释放资源
isr.close();
}
}
练习:转换文档的编码格式, 将GBK编码的文本文件转换为UTF-8编码的文本文档
* 练习:转换文件编码
将GBK编码的文本文件,转换为UTF-8编码的文本文件
* 分析:
1. 创建InputStreamReader对象,构造方法中传递字节输入流和指定的编码表名称GBK
2. 创建OutputStreamReader对象,构造方法中传递字节输出流和指定的编码表名称UTF-8
3. 使用InputStreamReader对象中的方法read读取文件
4. 使用OutputStreamReader对象中的方法write,把读取的数据写入到文件中
5. 释放资源
import java.io.*;
public class Demo04Test {
public static void main(String[] args) throws IOException {
//1. 创建InputStreamReader对象,构造方法中传递字节输入流和指定的编码表名称GBK
InputStreamReader isr = new InputStreamReader(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\我是GBK格式的文档.txt"),"GBK");
//2. 创建OutputStreamReader对象,构造方法中传递字节输出流和指定的编码表名称UTF-8
OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\我是UTF-8格式的文档.txt"),"UTF-8");
//3. 使用InputStreamReader对象中的方法read读取文件
int len = 0;
while((len = isr.read())!=-1){
//4. 使用OutputStreamReader对象中的方法write,把读取的数据写入到文件中
osw.write(len);
}
//5. 释放资源
osw.close();
isr.close();
}
}
四、序列化与反序列化
4.1 ObjectOutputStream序列化流,序列化–对象转换为字节
作用:把对象以流的方式写入到文件中保存
java.io.ObjectOutputStream extends OutputStream
ObjectOutputStream:对象的序列化流
* 作用:把对象以流的方式写入到文件中保存
* 构造方法:
ObjectOutputStream(OutputStream out) 创建写入指定 OutputStream 的 ObjectOutputStream
参数:
OutputStream out:字节输出流
* 特有的成员方法:
void writeObject(Object obj) 将指定的对象写入 ObjectOutputStream。
* 使用步骤:
1. 创建ObjectOutputStream对象,构造方法中传递字节输出流
2. 使用ObjectOutputStream中的方法writeObject,把对象写入到文件中
3. 资源释放
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutput;
import java.io.ObjectOutputStream;
public class Demo01ObjectOutputStream {
public static void main(String[] args) throws IOException {
//1. 创建ObjectOutputStream对象,构造方法中传递字节输出流
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\person.txt"));
//2. 使用ObjectOutputStream中的方法writeObject,把对象写入到文件中
oos.writeObject(new Person("迪丽热巴",18));
//3. 资源释放
oos.close();
}
}
4.2 序列化标记 Serializable
* 序列化和反序列化的时候,会抛出NotSerializableException没有序列化异常
* 类通过实现 java.io.Serializable接口以启动其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。
* Serializable接口也叫标记型接口
- 要进行序列化和反序列化的类必须实现Serializable接口,就会给类添加一个标记
- 当我们进行序列化和反序列化的时候,就会检测类上是否有这个标记
有:就可以序列化和反序列化
没有:就会抛出NotSerializableException异常
* 去市场上买肉-->肉上有一个蓝色章(检测合格)-->放心购买-->买回来怎么吃随意。
import java.io.Serializable;
public class Person implements Serializable{
private String name;
private int age;
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
4.3 ObjectInputStream反序列化流,反序列化–字节重构为对象
java.io.ObjectStream extends InputStream
ObjectInputStream:对象的反序列化流
* 作用:把文件中保存的对象,以流的方式读取出来使用
* 构造方法:
ObjectInputStream(InputStream in) 创建从指定 InputStream 读取的 ObjectInputStream。
参数:
InputStream in:字节输入流
* 特有的成员方法:
Object readObject() 从 ObjectInputStream 读取对象。
* 使用步骤:
1. 创建ObjectInputStream对象,构造方法中传递字节输入流
2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
3. 释放资源
4. 使用读取出来的对象(打印)
readObject方法声明抛出了ClassNotFoundException(class文件找不到异常)
当不存在对象的Java class文件时抛出此异常
反序列化的前提:
1. 类必须实现Serializable
2. 必须存在类对应的Java class文件
import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
public class Demo02ObjectInputStream {
public static void main(String[] args) throws IOException, ClassNotFoundException {
//1. 创建ObjectInputStream对象,构造方法中传递字节输入流
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\person.txt"));
//2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
Object o = ois.readObject();
//3. 释放资源
ois.close();
//4. 使用读取出来的对象(打印)
System.out.println(o);
Person p = (Person)o;
System.out.println(p.getName()+p.getAge());
}
}
打印输出:person.txt里的内容
4.4 被transient关键字修饰的成员变量不能被序列化
* 序列化和反序列化的时候,会抛出NotSerializableException没有序列化异常
* 类通过实现 java.io.Serializable接口以启动其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。
* Serializable接口也叫标记型接口
- 要进行序列化和反序列化的类必须实现Serializable接口,就会给类添加一个标记
- 当我们进行序列化和反序列化的时候,就会检测类上是否有这个标记
有:就可以序列化和反序列化
没有:就会抛出NotSerializableException异常
* 去市场上买肉-->肉上有一个蓝色章(检测合格)-->放心购买-->买回来怎么吃随意。
* static关键字:静态关键字
静态优先于非静态加载到内存中(静态优先于对象进入到内存中)
被static修饰的成员变量是不能被序列化的,序列化的都是对象。
private static int age;
oos.writeObject(new Person("迪丽热巴",18));
Object o = ois.readObject();
Person{name = '迪丽热巴',age = 0}
* transient关键字:瞬态关键字
被transient修饰的成员变量,不能被序列化
private transient int age;
oos.writeObject(new Person("迪丽热巴",18));
Object o = ois.readObject();
Person{name = '迪丽热巴',age = 0}
import java.io.Serializable;
public class Person implements Serializable{
private String name;
private int age;
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
4.5 序列化冲突异常的解决方案–> private static final long serialVersionUID = 42L;
指定序列号,不管类是否被修改,类的序列号都不会发生改变。
* 序列化和反序列化的时候,会抛出NotSerializableException没有序列化异常
* 类通过实现 java.io.Serializable接口以启动其序列化功能。未实现此接口的类将无法使其任何状态序列化或反序列化。
* Serializable接口也叫标记型接口
- 要进行序列化和反序列化的类必须实现Serializable接口,就会给类添加一个标记
- 当我们进行序列化和反序列化的时候,就会检测类上是否有这个标记
有:就可以序列化和反序列化
没有:就会抛出NotSerializableException异常
* 去市场上买肉-->肉上有一个蓝色章(检测合格)-->放心购买-->买回来怎么吃随意。
* static关键字:静态关键字
静态优先于非静态加载到内存中(静态优先于对象进入到内存中)
被static修饰的成员变量是不能被序列化的,序列化的都是对象。
private static int age;
oos.writeObject(new Person("迪丽热巴",18));
Object o = ois.readObject();
Person{name = '迪丽热巴',age = 0}
* transient关键字:瞬态关键字
被transient修饰的成员变量,不能被序列化
private transient int age;
oos.writeObject(new Person("迪丽热巴",18));
Object o = ois.readObject();
Person{name = '迪丽热巴',age = 0}
import java.io.Serializable;
public class Person implements Serializable{
private static final long serialVersionUID = 42L; //加上static final关键字,不管Person类是否发生修改,类的序列号都不能发生改变,这就避免了序列号冲突
private String name;
//private transient int age;
private int age;
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
public Person() {
}
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
}
* java.io.ObjectOutputStream extends OutputStream
* ObjectOutputStream:对象的序列化流
* 作用:把对象以流的方式写入到文件中保存
* 构造方法:
ObjectOutputStream(OutputStream out) 创建写入指定 OutputStream 的 ObjectOutputStream
参数:
OutputStream out:字节输出流
* 特有的成员方法:
void writeObject(Object obj) 将指定的对象写入 ObjectOutputStream。
* 使用步骤:
1. 创建ObjectOutputStream对象,构造方法中传递字节输出流
2. 使用ObjectOutputStream中的方法writeObject,把对象写入到文件中
3. 资源释放
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.ObjectOutput;
import java.io.ObjectOutputStream;
public class Demo01ObjectOutputStream {
public static void main(String[] args) throws IOException {
//1. 创建ObjectOutputStream对象,构造方法中传递字节输出流
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\person.txt"));
//2. 使用ObjectOutputStream中的方法writeObject,把对象写入到文件中
oos.writeObject(new Person("迪丽热巴",18));
//3. 资源释放
oos.close();
}
}
* java.io.ObjectStream extends InputStream
* ObjectInputStream:对象的反序列化流
* 作用:把文件中保存的对象,以流的方式读取出来使用
* 构造方法:
ObjectInputStream(InputStream in) 创建从指定 InputStream 读取的 ObjectInputStream。
参数:
InputStream in:字节输入流
* 特有的成员方法:
Object readObject() 从 ObjectInputStream 读取对象。
* 使用步骤:
1. 创建ObjectInputStream对象,构造方法中传递字节输入流
2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
3. 释放资源
4. 使用读取出来的对相关(打印)
readObject方法声明抛出了ClassNotFoundException(class文件找不到异常)
当不存在对象的class文件时抛出此异常
反序列化的前提:
1. 类必须实现Serializable
2. 必须存在类对应的class文件
import java.io.FileInputStream;
import java.io.IOException;
import java.io.ObjectInputStream;
public class Demo02ObjectInputStream {
public static void main(String[] args) throws IOException, ClassNotFoundException {
//1. 创建ObjectInputStream对象,构造方法中传递字节输入流
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\person.txt"));
//2. 使用ObjectInputStream对象中的方法readObject读取保存对象的文件
Object o = ois.readObject();
//3. 释放资源
ois.close();
//4. 使用读取出来的对相关(打印)
System.out.println(o);
Person p = (Person)o;
System.out.println(p.getName()+p.getAge());
}
}
4.6 练习:定义一个数组,把多个对象存储到一个集合中,并对集合进行序列化和反序列化
* 练习:序列化集合
当我们想在文件中保存多个对象的时候
可以把多个对象存储到一个集合中
对集合进行序列化和反序列化
* 分析:
1. 定义一个存储Person对象的ArrayList集合
2. 往ArrayList集合中存储Person对象
3. 创建一个序列化流ObjectOutputStream对象
4. 使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
5. 创建一个反序列化ObjectInputStream对象
6. 使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
7. 把Object类型的集合转换为ArrayList类型
8. 遍历ArrayList集合
9. 释放资源
import java.io.*;
import java.util.ArrayList;
public class Demo03Test {
public static void main(String[] args) throws IOException, ClassNotFoundException {
//1. 定义一个存储Person对象的ArrayList集合
ArrayList<Person> list = new ArrayList<>();
//2. 往ArrayList集合中存储Person对象
list.add(new Person("张三", 18));
list.add(new Person("李四", 19));
list.add(new Person("王五", 20));
//3. 创建一个序列化流ObjectOutputStream对象
ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("C:\\Users\\32189\\Desktop\\考研资料\\list.txt"));
//4. 使用ObjectOutputStream对象中的方法writeObject,对集合进行序列化
oos.writeObject(list);
//5. 创建一个反序列化ObjectInputStream对象
ObjectInputStream ois = new ObjectInputStream(new FileInputStream("C:\\Users\\32189\\Desktop\\考研资料\\list.txt"));
//6. 使用ObjectInputStream对象中的方法readObject读取文件中保存的集合
Object o = ois.readObject();
//7. 把Object类型的集合转换为ArrayList类型
ArrayList<Person> list2 = (ArrayList<Person>) o;
//8. 遍历ArrayList集合
for (Person p : list2) {
System.out.println(o);
}
//9. 释放资源
ois.close();
oos.close();
}
}
五、打印流,利用 PrintStream 指定打印输出到某指定文件中
* 可以改变输出语句的目的地(打印流的流向)
* 输出语句,默认在控制台输出
* 使用System.setOut方法改变输出语句的目的地改为参数中传递的打印流的目的地
static void setOut(PrintSteam out)
重新分配“标准”输出流
import java.io.FileNotFoundException;
import java.io.PrintStream;
public class Demo02PrintStream {
public static void main(String[] args) throws FileNotFoundException {
System.out.println("我是在控制台输出");
PrintStream ps = new PrintStream("C:\\Users\\32189\\Desktop\\考研资料\\目的地是打印流.txt");
System.out.println(ps); //把输出语句的目的地改变为打印流的目的地(.txt文档)
System.out.println("我在打印流的目的地中输出");
ps.close();
}
}