Java Day21

第一章 缓冲流

1.1 概述

缓冲流也叫做高效流,是对4个基本的FileXxx流的增强,所以也是4个类,按照数据类型分类:

  • 字节缓冲流:BufferedInputStream、BufferedOutputStream
  • 字符缓冲流:BufferedReader、BufferedWriter

缓冲流的基本原理,是在创建流对象时,会创建一个内置的默认大小的缓冲区数组,通过缓冲区读写,减少系统IO次数,从而提高读写的效率。

1.2 字节缓冲流

1.2.1 构造方法

  • public BufferedInputStream(InputStream in):创建一个新的缓冲输入流。
  • public BufferedOutputStream(OutputStream out):创建一个新的缓冲输出流。

1.2.2 代码演示:

  • 缓冲输出流
/*
* 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、创建BufferedOutputStream对象,构造方法中传入FileOutputStream对象,提高FileOutputStream对象效率
*   3、使用BufferedOutputStream对象中的方法write,把数据写入到内部缓冲区中。
*   4、使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
*   5、释放资源(会先调用flush方法刷新数据,第四步可以省略)
*
* */


import java.io.BufferedOutputStream;
import java.io.FileOutputStream;
import java.io.IOException;

public class Dmeo01BufferedOutputStream {
    public static void main(String[] args) throws IOException {
        // 创建FileOutputStream对象,构造方法中绑定要输出的目的地。
        FileOutputStream fos = new FileOutputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/a.txt");

        // 创建BufferedOutputStream对象,构造方法中传入FileOutputStream对象,提高FileOutputStream对象效率
        BufferedOutputStream bos = new BufferedOutputStream(fos);

        // 使用BufferedOutputStream对象中的方法write,把数据写入到内部缓冲区中。
        bos.write("我把数据写入到内部缓冲区中".getBytes());

        // 使用BufferedOutputStream对象中的方法flush,把内部缓冲区中的数据,刷新到文件中
        bos.flush();

        // 释放资源(会先调用flush方法刷新数据,第四步可以省略)
        bos.close();



    }
}
  • 缓冲输入流
/*
* java.io.BufferedInputStream extends InputStream
* BufferedInputStream:字节缓冲输入流
* 继承自父类的成员方法:
*   int read():从输入流中读取数据的下一个字节
*   int read(bytes[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b中,
*   void close() 关闭此输入流并释放与该流关联的所有系统资源。
*
* 构造方法:
*   BufferedInputStream(InputStream in) 创建一个BufferedInputStream并保存其参数,及输入流in,以便将来使用
*   BufferedInputStream(InputStream in, in size) 创建具有指定缓冲区大小的BufferedInputStream并保存其参数,即输入流in,以便将来使用
*   参数:
*       InputStream in:字节输入流
*           我们可以传递FileInputStream,缓冲流会给FileInputStream增加一个缓冲区大小时,提高FileInputStream的读取效率
*       int size:指定缓冲流内部缓冲区的大小,不指定默认
*
* 使用步骤:
*   1、创建FileInputStream对象,构造方法中绑定要读取的数据源
*   2、创建BufferedInputStream对象,构造方法中传递FileInputStream对象,提高FileInputStream对象的读取效率
*   3、使用BufferedInputStream对象中的方法read,读取该文件
*   4、释放资源
*
* */

import java.io.BufferedInputStream;
import java.io.FileInputStream;
import java.io.IOException;
import java.util.Arrays;

public class Demo02BufferedInputStream {
    public static void main(String[] args) throws IOException {
        // 1、创建FileInputStream对象,构造方法中绑定要读取的数据源
        FileInputStream fis = new FileInputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/a.txt");

        // 2、创建BufferedInputStream对象,构造方法中传递FileInputStream对象,提高FileInputStream对象的读取效率
        BufferedInputStream bif = new BufferedInputStream(fis);

        // 使用BufferedInputStream对象中的方法read,读取该文件
        // int read():从输入流中读取数据的下一个字节
//        int len = 0;  // 记录每次读取到的字节
//        while ((len = bif.read()) != -1) {
//            System.out.println(len);
//        }

        // int read(bytes[] b) 从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b中,
        int len = 0;  // 记录每次读取的有效字节个数
        byte[] bytes = new byte[1024];
        while ((len = bif.read(bytes)) != -1) {
            System.out.println(new String(bytes, 0, len));
        }

        bif.close();

    }
}

1.3 字符缓冲流

1.3.1 构造方法

  • public BufferedReader(Reader in):创建一个新的缓冲输入流。
  • public BufferedWriter(Writer out):创建一个新的缓冲输出流。

1.3.2 特有方法

  • BufferedReader:public String readLine():读取一行文字。
  • BufferedWriter:public void newLine():写一行行分隔符,由系统属性定义符号。

1.3.3 代码演示

  • 缓冲输出流
/*
* java.io.BufferedWriter extends Writer
* BUfferedWriter:字符缓冲输出流
* 继承自父类的共性成员方法:
*   void write(int c) 写入单个字符
*   void write(char[] cbuf) 写入字符数组
*   abstract void write(char[] cbuf, int off, int len) 写入字节数组的某一部分,off数组的开始索引,len写的字符个数
*   void write(String str) 写入字符串
*   void write(String str, int off, int len) 写入字符出啊的某一个部分,off是字符串的开始索引,len写的字符个数
*   void flush() 刷新该流的缓冲
*   void close() 关闭此流,但要先刷新它
*
* 构造方法:
*   BufferedWriter(Writer out) 创建一个使用默认大小输出缓冲区的缓冲字符输出流
*   BufferedWriter(Writer out, int sz) 创建一个使用给定大小输出缓冲区的新缓冲字符输出流
* 参数:
*   writer out:字符输出流,我们可以传递FileWriter,缓冲流会给FileWriter增加一个缓冲区,提高FileWriter的写入效率
*   int sz:指定缓冲区的大小,不写默认大小
*
* 特有的成员方法:
*   void newLine() 写入一个行分隔符,会根据不同的操作系统,获取不同的行分隔符
*
* 使用步骤:
*   创建字符缓冲流输出该对象,构造方法中传递字符输出流
*   调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
*   调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
*   释放资源
*
* */

import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;

public class Dmeo01BufferedWriter {
    public static void main(String[] args) throws IOException {
        // 创建字符缓冲流输出该对象,构造方法中传递字符输出流
        BufferedWriter bw = new BufferedWriter(new FileWriter("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/a.txt"));

        // 调用字符缓冲输出流中的方法write,把数据写入到内存缓冲区中
        for (int i = 0; i < 10; i++) {
            bw.write("你好,我是Tiger" + i + "号!");
            bw.newLine();
        }

        // 调用字符缓冲输出流中的方法flush,把内存缓冲区中的数据,刷新到文件中
        bw.flush();

        // 释放资源
        bw.close();


    }
}
  • 缓冲输入流
/*
* java.io.BufferedReader extends Reader
*
* 继承自父类的共性成员方法
*   int read() 读取单个字符并返回
*   int read(cahr[] cbuf) 一次读取多个字符,将字符读入数组
*   void close() 关闭该流并释放资源
*
* 构造方法:
*   BufferedReader(Reader in) 创建一个使用默认大小输入缓冲区的缓冲字符输入流。
*   BufferedReader(Reader in, int sz) 创建一个使用指定大小输入缓冲区的缓冲字符输入流
* 参数:
*   Reader in:字符输入流
*       我们可以传递FileReader,缓冲流会给FileReader增加一个缓冲区,提高FileReader的读取效率
* 特有的成员方法
*   String readLine() 读取一个文本行。读取一行数据,以行的终止符号来确认行结束
*   返回值:
*       包含该行内容的字符串,不包含任何终止符,如果已到达末尾,则返回null
* 使用步骤
*   1、创建字符缓冲输入流对象,构造方法中传递字符输入流
*   2、使用字符缓冲输入流对象中的方法read/readLine读取文本
*   3、释放资源
*
* */

import java.io.BufferedReader;
import java.io.FileReader;
import java.io.IOException;

public class Demo01BufferedReader {
    public static void main(String[] args) throws IOException {
        // 创建字符缓冲输入流对象,构造方法中传递字符输入流
        BufferedReader br = new BufferedReader(new FileReader("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/a.txt"));

        // 使用字符缓冲输入流对象中的方法read/readLine读取文本
        String line;
        while ((line = br.readLine()) != null) {
            System.out.println(line);
        }

        // 释放资源
        br.close();


    }
}

1.4 文本排序练习

讲下列文本按照开头序号排序

3.侍中、侍郎郭攸之、费祎、董允等,此皆良实,志虑忠纯,是以先帝简拔以遗陛下。愚以为宫中之事,事无大小,悉以咨之,然后施行,必得裨补阙漏,有所广益。
8.愿陛下托臣以讨贼兴复之效,不效,则治臣之罪,以告先帝之灵。若无兴德之言,则责攸之、祎、允等之慢,以彰其咎;陛下亦宜自谋,以咨诹善道,察纳雅言,深追先帝遗诏,臣不胜受恩感激。
4.将军向宠,性行淑均,晓畅军事,试用之于昔日,先帝称之曰能,是以众议举宠为督。愚以为营中之事,悉以咨之,必能使行阵和睦,优劣得所。
2.宫中府中,俱为一体,陟罚臧否,不宜异同。若有作奸犯科及为忠善者,宜付有司论其刑赏,以昭陛下平明之理,不宜偏私,使内外异法也。
1.先帝创业未半而中道崩殂,今天下三分,益州疲弊,此诚危急存亡之秋也。然侍卫之臣不懈于内,忠志之士忘身于外者,盖追先帝之殊遇,欲报之于陛下也。诚宜开张圣听,以光先帝遗德,恢弘志士之气,不宜妄自菲薄,引喻失义,以塞忠谏之路也。
9.今当远离,临表涕零,不知所言。
6.臣本布衣,躬耕于南阳,苟全性命于乱世,不求闻达于诸侯。先帝不以臣卑鄙,猥自枉屈,三顾臣于草庐之中,咨臣以当世之事,由是感激,遂许先帝以驱驰。后值倾覆,受任于败军之际,奉命于危难之间,尔来二十有一年矣。
7.先帝知臣谨慎,故临崩寄臣以大事也。受命以来,夙夜忧叹,恐付托不效,以伤先帝之明,故五月渡泸,深入不毛。今南方已定,兵甲已足,当奖率三军,北定中原,庶竭驽钝,攘除奸凶,兴复汉室,还于旧都。此臣所以报先帝而忠陛下之职分也。至于斟酌损益,进尽忠言,则攸之、祎、允之任也。
5.亲贤臣,远小人,此先汉所以兴隆也;亲小人,远贤臣,此后汉所以倾颓也。先帝在时,每与臣论此事,未尝不叹息痛恨于桓、灵也。侍中、尚书、长史、参军,此悉贞良死节之臣,愿陛下亲之信之,则汉室之隆,可计日而待也。

代码如下:

import java.io.*;
import java.util.HashMap;
import java.util.Set;

public class Dmeo01Main {
    public static void main(String[] args) throws IOException {
        BufferedReader br = new BufferedReader(new FileReader("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/TextSort/old.txt"));
        BufferedWriter bw = new BufferedWriter(new FileWriter("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/TextSort/new.txt"));

        String line;
        HashMap hashMap = new HashMap<>();
        while ((line = br.readLine()) != null) {
            char key = line.charAt(0);
            hashMap.put(key, line);
        }

        Set set = hashMap.keySet();
        for (char key : set) {
            bw.write(hashMap.get(key));
            bw.newLine();
            bw.flush();
        }

        br.close();
        bw.close();

    }
}

第二章 转换流

2.1 字符编码和字符集

2.1.1 字符编码

计算机中存储的信息都使用二进制表示的,而在我们屏幕上看到的数字、英文、标点、汉字等字符是二进制数转换之后的结果。按照某种规则,将字符存储到计算机中,称为编码。反之,将存储在计算机中的二进制按照某种规则解析显示出来,称为解码

  • 字符编码:就是一套自然语言的字符与二进制之间的对应规则。

2.1.2 字符集

  • 字符集:也叫编码集。是一个系统支持的所有字符的集合,包括各国的文字、标点符号、圆形符号、数字等。

计算机要准确的存储和识别各种字符集符号,需要进行编码,一套字符集必然至少有一套字符编码。常见的字符集有ASCII字符集、GBK字符集、Unicode字符集等。

Java Day21_第1张图片

2.2 编码引出的问题

在IDEA中,使用FileReader读取项目中的文本文件。由于IDEA的设置,都是默认的UTF-8编码,所以没有任何问题。但是,当读取windows系统中创建的文本文件的时候,由于windows系统的默认是GBK编码,就会出现乱码。

2.3 InputStreamReader类

转换流java.io.InputStreamReader,是Reader的子类,是从字节流到字符流的桥梁。它读取字节,并使用指定的字符集将其解码为字符。他的字符集可以指定,也可以接受平台的默认字符集。

2.3.1 构造方法

  • InputStreamReader(InputStream in):创建一个使用默认字符集的字符流。
  • InputStreamReader(InputStream in, String charsetName):创建一个指定字符集的字符流。

2.3.2 指定编码读取

/*
* java.io.InputStreamReader exxtends 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:指定的编码表名称,不区分大小写。
*
* 使用步骤:
*   1、创建InputStreamReader对象,构造方法中传递字节输入流和指定的编码表名称
*   2、使用InputStreamReader对象中的方法read读取文件
*   3、释放资源
*
* 注意事项:
*   构造方法中指定的编码表名称要和文件的编码相同,否则会发生乱码
*
*
* */

import java.io.FileInputStream;
import java.io.IOException;
import java.io.InputStreamReader;

public class Demo03InputStreamReader {
    public static void main(String[] args) throws IOException {
        // read_utr_8();
        read_gbk();

    }



    /*
    * 使用InputStreamReader读取GBK格式的文件
    * */
    private static void read_gbk() throws IOException {
        InputStreamReader isr = new InputStreamReader(new FileInputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/gbk.txt"), "gbk");

        int len = 0;
        while ((len = isr.read()) != -1) {
            System.out.println((char) len);
        }
        isr.close();
    }

    /*
    * 使用InputStreamReader读取UTF-8格式的文件
    * */
    private static void read_utr_8() throws IOException {
        // 创建InputStreamReader对象,构造方法中传递字节输入流和指定的编码表名称
        InputStreamReader isr = new InputStreamReader(new FileInputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/utf-8.txt"), "utf-8");

        // 使用InputStreamReader对象中的方法read读取文件
        int len = 0;
        while ((len = isr.read()) != -1) {
            System.out.println((char) len);
        }

        // 释放资源
        isr.close();

    }
}

2.4 OutputStreamWriter类

转换流java.io.OutputStreamWriter,是Writer的子类,是从字符流到字节流的桥梁。使用指定的字符集将字符编码为字节,他的字符集可以指定,也可以接受平台默认字符集。

2.4.1 构造方法

  • OutputStreamWriter(OutputStream in):创建一个使用默认字符集的字符流。
  • OutputStreamWriter(OutputStream in, String charsetName):创建一个指定字符集的字符流。

2.4.2 指定编码写出

/*
* 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 write(String strm int off, int len) 写入字符串的某一部分,off字符串的开始索引,len写的字符个数。
*   void flush() 刷新该流的缓冲
*   void close() 关闭此流,但要先刷新它
*
* 构造方法:
*   OutputStreamWriter(OutputStream out) 创建使用默认字符编码的OutputStreamWriter
*   OutputStreamWriter(OutputStream out, String charsetName) 创建使用指定字符集的 OutputStreamWriter
*   参数:
*       OutputStram out:字节输出流,可以用来写转换之后的字节到文件中
*       String charsetName:指定的编码表名称,不区分大小写,可以是utf-8/GBK...等,不指定默认使用UTF-8
*
* 使用步骤:
*   1、创建OutputStreamWriter对象,构造方法中传递字节输出流和指定的编码表名称
*   2、使用OutputStreamWriter对象中的方法write,把字符转换为字节存储缓冲区中
*   3、使用OutputStreamWriter对象中的方法flush,把内存缓冲区中的字节刷新到文件中
*   4、释放资源
* */

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;

public class Demo02OutPutStreamWriter {
    public static void main(String[] args) throws IOException {
        // write_utf_8();
        write_gbk();
    }

    /*
    * 使用转换流OutputStreamWriter写GBK格式的文件
    *
    * */
    private static void write_gbk() throws IOException {
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/gbk.txt"), "gbk");
        osw.write("你好");
        osw.flush();
        osw.close();

    }

    /*
    * 使用转换流OutputStreamWriter写UTF-8格式的文件
    *
    * */
    private static void write_utf_8() throws IOException {
        // 创建OutputStreamWriter对象,构造方法中传递字节输出流和指定的编码表名称
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/utf-8.txt"), "utf-8");

        // 使用OutputStreamWriter对象中的方法write,把字符转换为字节存储缓冲区中
        osw.write("你好");

        // 使用OutputStreamWriter对象中的方法flush,把内存缓冲区中的字节刷新到文件中
        osw.flush();

        // 释放资源
        osw.close();
    }

}

2.4.3 转换流理解图解

Java Day21_第2张图片

2.5 练习:转换文件编码

将上面的《出师表》的编码转换

代码演示:

import java.io.*;

public class Demo04TestCodeReverse {
    public static void main(String[] args) throws IOException {
        OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/new.txt"), "gbk");
        InputStreamReader isr = new InputStreamReader(new FileInputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ReverseStream/old.txt"), "utf-8");
        int len = 0;
        char[] chars = new char[1024];
        while ((len = isr.read(chars)) != -1) {
            osw.write(chars);
            osw.write("\n");
            osw.flush();
        }
        isr.close();
        osw.close();
    }

}

第三章 序列化

3.1 概述

java提供了一种对象序列化的机制。用一个字节序列可以表示一个对象,该字节序列包含该对象的数值、对象的类型和对象中存储的属性等信息。字节序列写出到文件之后,相当于文件中持久保存了一个对象的信息。反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化。对象的数据、对象的类型和对象中存储的数据信息,都可以用来在内存中创建对象,如下图:

Java Day21_第3张图片

3.2 ObjectOutputStream类

java.io.ObjectOutputStream类,将java对象的原始数据类型写出到文件,实现对象的持久存储。

3.2.1 构造方法

  • java.io.ObjectOutputStream类:将java对象的原始数据类型写出到文件,实现对象的持久存储。

3.2.3 序列化操作

  • 一个对象想要序列化,必须满足两个条件
    • 该类必须实现了java.io.Serializable接口,Serializable是一个标记接口,不实现此接口的类将不会使任何状态的序列化和反序列化,会抛出NotSerializableException。
    • 该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用transient关键字修饰。
  • 写出对象方法:
    • public final void writeObject(Object obj):将指定对象写出。
  • 代码演示:
/*
* 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.ObjectOutputStream;

public class Demo01ObjectOutputSream {
    public static void main(String[] args) throws IOException {
        // 创建ObjectOutputStream对象,构造方法中传递字节输出流
        ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ObjectStream/Person.txt"));

        // 使用ObjectOutputStream对象中的方法writeObject,把对象写入到文件中
        oos.writeObject(new Person("tiger", 18));

        // 释放资源
        oos.close();
    }
}

3.4 ObjectInputStream类

ObjectInputStream反序列化流,将之前使用ObjectInputStream序列化的原始数据恢复为对象。

3.4.1 构造方法

  • public ObjectInputStream(InputStream in):创建一个指定InputStream的ObjectInputStream。

3.4.2 反序列化操作

  • 如果能找到一个对象的class文件,我们进行反序列化操作,调用ObjectInputStream读取对象的方法。
    • public final Object readObject():读取一个对象。
  • 当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个InvalidClassException异常,发生这个异常的原因如下:
    • 该类的序列版本号与从流中读取的类型描述符的版本号不匹配
    • 该类包含位置数据类型
    • 该类没有可访问的无参构造方法
  • Serializable接口给需要序列化的类,提供了一个序列化版本号,serialVersion该版本号的目的在于验证序列化的对象和对应类是否版本匹配。

代码演示:

定义Person类:

/*
* 序列化和反序列化的时候,会抛出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关键字:瞬态关键字
*   被tansient修饰成员变量不能被序列化
*   private static int age;
*   oos.writeObject(new Person("小美女"), 18)
*   Object o = ois.readObject();
*   Person(name="小美女", age=0)
* */

import java.io.Serializable;

public class Person implements Serializable {
    // 定义这个常量用来保证当类发生变化的时候,反序列化产生的InvalidClassException异常,定义该常量后,seialVersionUID不会改变
    // 该常量必须是静态的最终的long类型字段,即static final long类型声明
    static final long seialVersionUID = 1L;
    private String name;
    // private int age;
    public int 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;
    }

    @Override
    public String toString() {
        return "Person{" +
                "name='" + name + '\'' +
                ", age=" + age +
                '}';
    }
}

定义测试类:

/*
* java.io.ObjectInputStream 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 {
        // 创建使用ObjectInputStream对象,构造方法中传递字节输入流
        ObjectInputStream ois = new ObjectInputStream(new FileInputStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/ObjectStream/Person.txt"));

        // 使用ObjectInputStream对象中的方法readObject 读取保存对象的文件
        Object o = ois.readObject();

        // 释放资源
        ois.close();

        // 使用读取出来的对象(打印)
        System.out.println(o);



    }

}
 

第四章 打印流

4.1 概述

平时我们在控制台打印输出,是调用print方法和println方法完成的,这两个方法都来自于java.io.PrintStream类,该类能够方便地打印各种数据类型的值,是一种便捷的输入方式。

4.2 PrintStream类

4.2.1 构造方法

  • public PrintStream(Stream fileName):使用指定的文件名创建一个新的打印流。
  • 代码演示
/*
* 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方法写数据,那么查看数据的时候会查询编码表
*   如果使用自己特有的方法print/println方法写数据,写的数据会原样输出
* */

import java.io.FileNotFoundException;
import java.io.PrintStream;

public class Demo01PrintStream {
    public static void main(String[] args) throws FileNotFoundException {
        // 创建打印流PrintStream对象,构造方法中绑定要输出的目的地
        PrintStream ps = new PrintStream("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/PrintStream/print.txt");


        // 如果使用继承自父类的write方法写数据,那么查看数据的时候会查询编码表
        ps.write(97);

        // 如果使用自己特有的方法print/println方法写数据,写的数据会原样输出
        ps.println(97);
        ps.println(8.8);
        ps.println("shishf");

        // 释放资源
        ps.close();
    }
}

4.2.2 改变打印流向

System.out就是一个PrintStream类型的,只不过它的流向是系统规定的,打印在控制台上。不过我们可以改变它的流向,代码如下:

/*
* 可以改变输出语句的目的地(打印流的语句)
* 输出语句,默认在控制台输出
* 使用System.setOut方法改变输出语句的目的地改为参数中传递的打印流的目的地
*   static void setOut(PrintStream 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("/Users/shiwenhu/IdeaProjects/advance-code/src/com/it/day10code/PrintStream/打印流的输出目的地.txt");
        System.setOut(ps);
        System.out.println("我在打印流的目的地输出");

        ps.close();

    }
}

 

你可能感兴趣的:(java学习)