缓冲流、转换流、序列化流、打印流

一、字节缓冲流

在这里插入图片描述

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();
    }
}
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值