【缓冲流、转换流、序列化流、装饰者模式、commons-io工具包】

  • 第一章 缓冲流
    • 1.1字节缓冲流
    • 1.2字符缓冲流
    • 1.3实操--文本排序
  • 第二章 转换流
    • 2.1字符编码和字符集
    • 2.2编码引出的问题
    • 2.3InputStreamReader类
    • 2.4OutputStreamWriter类
    • 2.5实操--转换文件编码
  • 第三章 序列化
    • 3.1序列化和反序列化的概念
    • 3.2ObjectOutputStream类
    • 3.3ObjectInputStream类
    • 3.4序列化和反序列化注意事项
    • 3.5实操--序列化集合
  • 第四章 打印流
    • 4.1打印流的概述
    • 4.2 打印流的使用
  • 第五章 装饰设计模式
    • 5.1装饰模式概述
    • 5.2装饰模式遵循原则:
  • 第六章 commons-io工具包
  • 总结
  • 练习模块
    • 题目一
    • 第二题
    • 第三题
    • 第四题

第一章 缓冲流

昨天学习了基本的一些流,作为IO流的入门,今天我们要见识一些更强大的流。比如能够高效读写的缓冲流,能够转换编码的转换流,能够持久化存储对象的序列化流等等。这些功能更为强大的流,都是在基本的流对象基础之上创建而来的,就像穿上铠甲的武士一样,相当于是对基本流对象的一种增强。

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

  • 字节缓冲流BufferedInputStreamBufferedOutputStream
  • 字符缓冲流BufferedReaderBufferedWriter

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

1.1字节缓冲流

字节缓冲流的构造方法

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

构造举例,代码如下:

// 创建字节缓冲输入流
BufferedInputStream bis = new BufferedInputStream(new FileInputStream("bis.txt"));
// 创建字节缓冲输出流
BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("bos.txt"));

拷贝文件效率测试

查询API,缓冲流读写方法与基本的流是一致的,我们通过复制大文件(375MB),测试它的效率。

  1. 基本流,代码如下:
public class BufferedDemo {public static void main(String[] args) throws FileNotFoundException {// 记录开始时间long start = System.currentTimeMillis();// 创建流对象try (FileInputStream fis = new FileInputStream("jdk8.exe");FileOutputStream fos = new FileOutputStream("copy.exe")){// 读写数据int b;while ((b = fis.read()) != -1) {fos.write(b);}} catch (IOException e) {e.printStackTrace();}// 记录结束时间long end = System.currentTimeMillis();System.out.println("普通流复制时间:"+(end - start)+" 毫秒");}
}十几分钟过去了...
  1. 缓冲流,代码如下:
public class BufferedDemo {public static void main(String[] args) throws FileNotFoundException {// 记录开始时间long start = System.currentTimeMillis();// 创建流对象try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk8.exe"));BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.exe"));){// 读写数据int b;while ((b = bis.read()) != -1) {bos.write(b);}} catch (IOException e) {e.printStackTrace();}// 记录结束时间long end = System.currentTimeMillis();System.out.println("缓冲流复制时间:"+(end - start)+" 毫秒");}
}缓冲流复制时间:8016 毫秒

如何更快呢?

使用数组的方式,代码如下:

public class BufferedDemo {public static void main(String[] args) throws FileNotFoundException {// 记录开始时间long start = System.currentTimeMillis();// 创建流对象try (BufferedInputStream bis = new BufferedInputStream(new FileInputStream("jdk8.exe"));BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("copy.exe"));){// 读写数据int len;byte[] bytes = new byte[8*1024];while ((len = bis.read(bytes)) != -1) {bos.write(bytes, 0 , len);}} catch (IOException e) {e.printStackTrace();}// 记录结束时间long end = System.currentTimeMillis();System.out.println("缓冲流使用数组复制时间:"+(end - start)+" 毫秒");}
}
缓冲流使用数组复制时间:666 毫秒

1.2字符缓冲流

字符缓冲流的构造方法

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

构造举例,代码如下:

// 创建字符缓冲输入流
BufferedReader br = new BufferedReader(new FileReader("br.txt"));
// 创建字符缓冲输出流
BufferedWriter bw = new BufferedWriter(new FileWriter("bw.txt"));

字符缓冲流的特有方法

字符缓冲流的基本方法与普通字符流调用方式一致,不再阐述,我们来看它们具备的特有方法。

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

readLine方法演示,代码如下:

public class BufferedReaderDemo {public static void main(String[] args) throws IOException {// 创建流对象BufferedReader br = new BufferedReader(new FileReader("in.txt"));// 定义字符串,保存读取的一行文字String line  = null;// 循环读取,读取到最后返回nullwhile ((line = br.readLine())!=null) {System.out.print(line);System.out.println("------");}// 释放资源br.close();}
}

newLine方法演示,代码如下:

public class BufferedWriterDemo throws IOException {public static void main(String[] args) throws IOException  {// 创建流对象BufferedWriter bw = new BufferedWriter(new FileWriter("out.txt"));// 写出数据bw.write("黑马");// 写出换行bw.newLine();bw.write("程序");bw.newLine();bw.write("员");bw.newLine();// 释放资源bw.close();}
}
输出效果:
黑马
程序
员

1.3实操–文本排序

需求

请将文本信息恢复顺序。

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

分析

  1. 逐行读取文本信息。
  2. 解析文本信息到集合中。
  3. 遍历集合,按顺序,写出文本信息。

实现

public class Test {public static void main(String[] args) throws Exception {// 1.创建一个ArrayList集合,用来存储读取到的每行数据ArrayList<String> list = new ArrayList<>();// 2.创建字符缓冲输入流对象FileReader fr = new FileReader("day17\\aaa\\c.txt");BufferedReader br = new BufferedReader(fr);// 3.定义一个String类型的变量,用来存储读取到的行数据String line;// 4.循环读取while ((line = br.readLine()) != null) {// 5.在循环中,把读取到的行数据作为元素添加到集合中list.add(line);}// 6.关闭流,释放资源fr.close();// 7.对集合进行排序Collections.sort(list); // 字符串的升序排序,String类型实现Comparable  compareTo方法: 指定的就是按照字典排序//  b  a  c   ---> 字典顺序   a  b  c// 8.创建字符缓冲输出流对象FileWriter fw = new FileWriter("day17\\aaa\\c.txt");// 清空文件BufferedWriter bw = new BufferedWriter(fw);// 9.对排行序的集合进行循环遍历for (String s : list) {// 10.在循环中,把遍历出来的行数据写出到c.txt文件中bw.write(s);bw.newLine();}// 11.关闭流,释放资源bw.close();}
}

第二章 转换流

2.1字符编码和字符集

字符编码的概述

计算机中储存的信息都是用二进制数表示的,而我们在屏幕上看到的数字、英文、标点符号、汉字等字符是二进制数转换之后的结果。按照某种规则,将字符存储到计算机中,称为编码 。反之,将存储在计算机中的二进制数按照某种规则解析显示出来,称为解码 。比如说,按照A规则存储,同样按照A规则解析,那么就能显示正确的文本f符号。反之,按照A规则存储,再按照B规则解析,就会导致乱码现象。

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

字符集的概述

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

计算机要准确的存储和识别各种字符集符号,需要进行字符编码,一套字符集必然至少有一套字符编码。常见字符集有ASCII字符集、GBK字符集、Unicode字符集等。
可见,当指定了编码,它所对应的字符集自然就指定了,所以编码才是我们最终要关心的。

  • ASCII字符集

    • ASCII(American Standard Code for Information Interchange,美国信息交换标准代码)是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)。
    • 基本的ASCII字符集,使用7位(bits)表示一个字符,共128字符。ASCII的扩展字符集使用8位(bits)表示一个字符,共256字符,方便支持欧洲常用字符
  • ISO-8859-1字符集
    • 拉丁码表,别名Latin-1,用于显示欧洲使用的语言,包括荷兰、丹麦、德语、意大利语、西班牙语等。
    • ISO-5559-1使用单字节编码,兼容ASCII编码。
  • GBxxx字符集
    • GB就是国标的意思,是为了显示中文而设计的一套字符集。
    • GB2312:简体中文码表。一个小于127的字符的意义与原来相同。但两个大于127的字符连在一起时,就表示一个汉字,这样大约可以组合了包含7000多个简体汉字,此外数学符号、罗马希腊的字母、日文的假名们都编进去了,连在ASCII里本来就有的数字、标点、字母都统统重新编了两个字节长的编码,这就是常说的"全角"字符,而原来在127号以下的那些就叫"半角"字符了。
    • GBK:最常用的中文码表。是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等。
    • GB18030:最新的中文码表。收录汉字70244个,采用多字节编码,每个字可以由1个、2个或4个字节组成。支持中国国内少数民族的文字,同时支持繁体汉字以及日韩汉字等。
  • Unicode字符集
    • Unicode编码系统为表达任意语言的任意字符而设计,是业界的一种标准,也称为统一码、标准万国码。
    • 它最多使用4个字节的数字来表达每个字母、符号,或者文字。有三种编码方案,UTF-8、UTF-16和UTF-32。最为常用的UTF-8编码。
    • UTF-8编码,可以用来表示Unicode标准中任何字符,它是电子邮件、网页及其他存储或传送文字的应用中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。所以,我们开发Web应用,也要使用UTF-8编码。它使用一至四个字节为每个字符编码,编码规则:
      1. 128个US-ASCII字符,只需一个字节编码。
      2. 拉丁文等字符,需要二个字节编码。
      3. 大部分常用字(含中文),使用三个字节编码。
      4. 其他极少使用的Unicode辅助字符,使用四字节编码。

2.2编码引出的问题

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

public class ReaderDemo {public static void main(String[] args) throws IOException {FileReader fileReader = new FileReader("E:\\File_GBK.txt");int read;while ((read = fileReader.read()) != -1) {System.out.print((char)read);}fileReader.close();}
}
输出结果:
���

那么如何读取GBK编码的文件呢?

2.3InputStreamReader类

InputStreamReader类的概述

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

InputStreamReader类的构造方法

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

构造举例,代码如下:

InputStreamReader isr = new InputStreamReader(new FileInputStream("in.txt"));
InputStreamReader isr2 = new InputStreamReader(new FileInputStream("in.txt") , "GBK");

InputStreamReader类指定编码读取

public class ReaderDemo2 {public static void main(String[] args) throws IOException {// 定义文件路径,文件为gbk编码String FileName = "E:\\file_gbk.txt";// 创建流对象,默认UTF8编码InputStreamReader isr = new InputStreamReader(new FileInputStream(FileName));// 创建流对象,指定GBK编码InputStreamReader isr2 = new InputStreamReader(new FileInputStream(FileName) , "GBK");// 定义变量,保存字符int read;// 使用默认编码字符流读取,乱码while ((read = isr.read()) != -1) {System.out.print((char)read); // ��Һ�}isr.close();// 使用指定编码字符流读取,正常解析while ((read = isr2.read()) != -1) {System.out.print((char)read);// 大家好}isr2.close();}
}

2.4OutputStreamWriter类

OutputStreamWriter类的概述

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

OutputStreamWriter类的构造方法

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

构造举例,代码如下:

OutputStreamWriter isr = new OutputStreamWriter(new FileOutputStream("out.txt"));
OutputStreamWriter isr2 = new OutputStreamWriter(new FileOutputStream("out.txt") , "GBK");

OutputStreamWriter类指定编码读取

public class OutputDemo {public static void main(String[] args) throws IOException {// 定义文件路径String FileName = "E:\\out.txt";// 创建流对象,默认UTF8编码OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream(FileName));// 写出数据osw.write("你好"); // 保存为6个字节osw.close();// 定义文件路径String FileName2 = "E:\\out2.txt";// 创建流对象,指定GBK编码OutputStreamWriter osw2 = new OutputStreamWriter(new FileOutputStream(FileName2),"GBK");// 写出数据osw2.write("你好");// 保存为4个字节osw2.close();}
}

转换流理解图解

转换流是字节与字符间的桥梁!

2.5实操–转换文件编码

需求

  • 将GBK编码的文本文件,转换为UTF-8编码的文本文件。

分析

  1. 指定GBK编码的转换流,读取文本文件。
  2. 使用UTF-8编码的转换流,写出文本文件。

实现

public class TransDemo {public static void main(String[] args) {      // 1.定义文件路径String srcFile = "file_gbk.txt";String destFile = "file_utf8.txt";// 2.创建流对象// 2.1 转换输入流,指定GBK编码Reader isr = new InputStreamReader(new FileInputStream(srcFile) , "GBK");// 2.2 转换输出流,默认utf8编码Writer osw = new OutputStreamWriter(new FileOutputStream(destFile));// 3.读写数据// 3.1 定义数组char[] cbuf = new char[1024];// 3.2 定义长度int len;// 3.3 循环读取while ((len = isr.read(cbuf))!=-1) {// 循环写出osw.write(cbuf,0,len);}// 4.释放资源osw.close();isr.close();}
}

第三章 序列化

3.1序列化和反序列化的概念

Java 提供了一种对象序列化的机制。用一个字节序列可以表示一个对象,该字节序列包含该对象的数据对象的类型对象中存储的属性等信息。字节序列写出到文件之后,相当于文件中持久保存了一个对象的信息。

反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化对象的数据对象的类型对象中存储的数据信息,都可以用来在内存中创建对象。看图理解序列化:

3.2ObjectOutputStream类

ObjectOutputStream类的概述

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

ObjectOutputStream类构造方法

  • public ObjectOutputStream(OutputStream out): 创建一个指定OutputStream的ObjectOutputStream。

构造举例,代码如下:

FileOutputStream fileOut = new FileOutputStream("employee.txt");
ObjectOutputStream out = new ObjectOutputStream(fileOut);

ObjectOutputStream类序列化操作

  1. 一个对象要想序列化,必须满足两个条件:
  • 该类必须实现java.io.Serializable 接口,Serializable 是一个标记接口
  • 该类的所有属性必须是可序列化的。
public class Person implements Serializable {private String name;private int age;public Animal anl;// 宠物public Person() {}public Person(String name, int age) {this.name = name;this.age = age;}//...set get ...toString...
}public class Animal implements Serializable {}

2.写出对象方法

  • public final void writeObject (Object obj) : 将指定的对象写出。
public class Test {public static void main(String[] args) throws Exception{// 使用ObjectOutputStream类序列化一个Person对象到文件中// 创建Person对象Person p = new Person("张三",18);Animal anl = new Animal();p.anl = anl;// 创建序列化流对象,关联目的地文件路径FileOutputStream fos = new FileOutputStream("day17\\ccc\\a.txt");ObjectOutputStream oos = new ObjectOutputStream(fos);// 写出对象oos.writeObject(p);// 关闭流,释放资源oos.close();}
}

3.3ObjectInputStream类

ObjectInputStream类的概述

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

ObjectInputStream类构造方法

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

ObjectInputStream类反序列化操作1

如果能找到一个对象的class文件,我们可以进行反序列化操作,调用ObjectInputStream读取对象的方法:

  • public final Object readObject () : 读取一个对象。
public class DeserializeDemo {public static void main(String [] args)   {Employee e = null;try {        // 创建反序列化流FileInputStream fileIn = new FileInputStream("employee.txt");ObjectInputStream in = new ObjectInputStream(fileIn);// 读取一个对象e = (Employee) in.readObject();// 释放资源in.close();fileIn.close();}catch(IOException i) {// 捕获其他异常i.printStackTrace();return;}catch(ClassNotFoundException c)  {// 捕获类找不到异常System.out.println("Employee class not found");c.printStackTrace();return;}// 无异常,直接打印输出System.out.println("Name: " + e.name);  // zhangsanSystem.out.println("Address: " + e.address); // beiqingluSystem.out.println("age: " + e.age); // 0}
}

3.4序列化和反序列化注意事项

序列化的注意事项

  • 该类必须实现java.io.Serializable 接口,Serializable 是一个标记接口,不实现此接口的类将不会使任何状态序列化或反序列化,会抛出NotSerializableException
  • 该类的所有属性必须是可序列化的。如果有一个属性不需要可序列化的,则该属性必须注明是瞬态的,使用transient 关键字修饰。

反序列化的注意事项

  • 对于JVM可以反序列化对象,它必须是能够找到class文件的类。如果找不到该类的class文件,则抛出一个 ClassNotFoundException 异常。

  • 另外,当JVM反序列化对象时,能找到class文件,但是class文件在序列化对象之后发生了修改,那么反序列化操作也会失败,抛出一个InvalidClassException异常。发生这个异常的原因如下:

    • 该类的序列版本号与从流中读取的类描述符的版本号不匹配
    • 该类包含未知数据类型
    • 该类没有可访问的无参数构造方法

Serializable 接口给需要序列化的类,提供了一个序列版本号。serialVersionUID 该版本号的目的在于验证序列化的对象和对应类是否版本匹配。

public class Employee implements java.io.Serializable {// 加入序列版本号private static final long serialVersionUID = 1L;public String name;public String address;// 添加新的属性 ,重新编译, 可以反序列化,该属性赋为默认值.public int eid; public void addressCheck() {System.out.println("Address  check : " + name + " -- " + address);}
}

3.5实操–序列化集合

需求

  1. 将存有多个自定义对象的集合序列化操作,保存到list.txt文件中。
  2. 反序列化list.txt ,并遍历集合,打印对象信息。

分析

  1. 把若干学生对象 ,保存到集合中。
  2. 把集合序列化。
  3. 反序列化读取时,只需要读取一次,转换为集合类型。
  4. 遍历集合,可以打印所有的学生信息

实现

public class SerTest {public static void main(String[] args) throws Exception {// 创建 学生对象Student student = new Student("老王", "laow");Student student2 = new Student("老张", "laoz");Student student3 = new Student("老李", "laol");ArrayList<Student> arrayList = new ArrayList<>();arrayList.add(student);arrayList.add(student2);arrayList.add(student3);// 序列化操作// serializ(arrayList);// 反序列化  ObjectInputStream ois  = new ObjectInputStream(new FileInputStream("list.txt"));// 读取对象,强转为ArrayList类型ArrayList<Student> list  = (ArrayList<Student>)ois.readObject();for (int i = 0; i < list.size(); i++ ){Student s = list.get(i);System.out.println(s.getName()+"--"+ s.getPwd());}}private static void serializ(ArrayList<Student> arrayList) throws Exception {// 创建 序列化流 ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("list.txt"));// 写出对象oos.writeObject(arrayList);// 释放资源oos.close();}
}

第四章 打印流

4.1打印流的概述

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

4.2 打印流的使用

  • public PrintStream(String fileName): 使用指定的文件名创建一个新的打印流。

构造举例,代码如下:

PrintStream ps = new PrintStream("ps.txt");

System.out就是PrintStream类型的,只不过它的流向是系统规定的,打印在控制台上。不过,既然是流对象,我们就可以玩一个"小把戏",将数据输出到指定文本文件中。

public class Test {public static void main(String[] args) throws Exception {/*打印流:PrintStream类 顶层父类 OutputStream类   字节输出流- 打印流的概述java.io.PrintStream类,该类能够方便地打印各种数据类型的值,是一种便捷的输出方式。- 打印流的使用创建打印流对象:public PrintStream(String fileName): 使用指定的文件名创建一个新的打印流。打印数据的方法:print(xxx)println(xxx)*/// 指定路径打印/*// 创建打印流对象PrintStream ps = new PrintStream("day17\\ddd\\a.txt");// 打印数据ps.println(100);ps.println("java");ps.println(true);// 关闭流,释放资源ps.close();*/// 系统的打印是默认打印到控制台System.out.println("java1");PrintStream ps = System.out;ps.println("java2");// 改变系统打印流打印数据的目的地PrintStream ps2 = new PrintStream("day17\\ddd\\b.txt");// 设置系统类的打印流对象为ps2System.setOut(ps2);System.out.println("java3");// 打印b.txt文件中}
}

第五章 装饰设计模式

5.1装饰模式概述

在我们今天所学的缓冲流中涉及到java的一种设计模式,叫做装饰模式,我们来认识并学习一下这个设计模式。

装饰模式指的是在不改变原类, 不使用继承的基础上,动态地扩展一个对象的功能。

5.2装饰模式遵循原则:

  1. 装饰类和被装饰类必须实现相同的接口
  2. 在装饰类中必须传入被装饰类的引用
  3. 在装饰类中对需要扩展的方法进行扩展
  4. 在装饰类中对不需要扩展的方法调用被装饰类中的同名方法

案例演示

准备环境:

  1. 编写一个Star接口, 提供sing 和 dance抽象方法
  2. 编写一个LiuDeHua类,实现Star接口,重写抽象方法
public interface Star {public void sing();public void dance();
}
public class LiuDeHua implements Star {@Overridepublic void sing() {System.out.println("刘德华在唱忘情水...");}@Overridepublic void dance() {System.out.println("刘德华在跳街舞...");}
}

需求:

​ 在不改变原类的基础上对LiuDeHua类的sing方法进行扩展

实现步骤:

  1. 编写一个LiuDeHuaWarpper类, 实现Star接口,重写抽象方法
  2. 提供LiuDeHuaWarpper类的有参构造, 传入LiuDeHua类对象
  3. 在LiuDeHuaWarpper类中对需要增强的sing方法进行增强
  4. 在LiuDeHuaWarpper类对不需要增强的方法调用LiuDeHua类中的同名方法

实现代码如下

LiuDeHua类: 被装饰类

LiuDeHuaWarpper类: 我们称之为装饰类

/*装饰模式遵循原则:装饰类和被装饰类必须实现相同的接口在装饰类中必须传入被装饰类的引用在装饰类中对需要扩展的方法进行扩展在装饰类中对不需要扩展的方法调用被装饰类中的同名方法
*/
public class LiuDeHuaWarpper implements Star {// 存放被装饰类的引用private LiuDeHua liuDeHua;// 通过构造器传入被装饰类对象public LiuDeHuaWarpper(LiuDeHua liuDeHua){this.liuDeHua = liuDeHua;}@Overridepublic void sing() {// 对需要扩展的方法进行扩展增强System.out.println("刘德华在鸟巢的舞台上演唱忘情水.");}@Overridepublic void dance() {// 不需要增强的方法调用被装饰类中的同名方法liuDeHua.dance();}
}

测试结果

public static void main(String[] args) {// 创建被装饰类对象LiuDeHua liuDeHua = new LiuDeHua();// 创建装饰类对象,被传入被装饰类LiuDeHuaWarpper liuDeHuaWarpper = new LiuDeHuaWarpper(liuDeHua);// 调用装饰类的相关方法,完成方法扩展liuDeHuaWarpper.sing();liuDeHuaWarpper.dance();
}

小结

装饰模式可以在不改变原类的基础上对类中的方法进行扩展增强,实现原则为:

  1. 装饰类和被装饰类必须实现相同的接口
  2. 在装饰类中必须传入被装饰类的引用
  3. 在装饰类中对需要扩展的方法进行扩展
  4. 在装饰类中对不需要扩展的方法调用被装饰类中的同名方法

第六章 commons-io工具包

commons-io工具包的概述

commons-io是apache开源基金组织提供的一组有关IO操作的类库,可以挺提高IO功能开发的效率。commons-io工具包提供了很多有关io操作的类,见下表:

功能描述
org.apache.commons.io 有关Streams、Readers、Writers、Files的工具类
org.apache.commons.io.input 输入流相关的实现类,包含Reader和InputStream
org.apache.commons.io.output 输出流相关的实现类,包含Writer和OutputStream
org.apache.commons.io.serialization 序列化相关的类

commons-io工具包的使用

步骤:

  1. 下载commons-io相关jar包;http://commons.apache.org/proper/commons-io/
  2. 把commons-io-2.6.jar包复制到指定的Module的lib目录中
  3. 将commons-io-2.6.jar加入到classpath中

commons-io工具包的使用

  • commons-io提供了一个工具类 org.apache.commons.io.IOUtils,封装了大量IO读写操作的代码。其中有两个常用方法:
  1. public static int copy(InputStream in, OutputStream out); 把input输入流中的内容拷贝到output输出流中,返回拷贝的字节个数(适合文件大小为2GB以下)
  2. public static long copyLarge(InputStream in, OutputStream out);把input输入流中的内容拷贝到output输出流中,返回拷贝的字节个数(适合文件大小为2GB以上)

文件复制案例演示:

 // IOUtils工具类拷贝文件private static void method01() throws IOException {FileInputStream fis = new FileInputStream("day17\\aaa\\jdk11.exe");FileOutputStream fos = new FileOutputStream("day17\\aaa\\jdk11Copy4.exe");IOUtils.copy(fis,fos);fos.close();fis.close();}
  • commons-io还提供了一个工具类org.apache.commons.io.FileUtils,封装了一些对文件操作的方法:
  1. public static void copyFileToDirectory(final File srcFile, final File destFile) //复制文件到另外一个目录下。
  2. public static void copyDirectoryToDirectory( file1 , file2 );//复制file1目录到file2位置。

案例演示:

public static void main(String[] args) throws IOException {// FileUtils工具类拷贝文件到指定文件夹// File srcFile = new File("day17\\aaa\\a.txt");// File destFile = new File("day17\\eee");// FileUtils.copyFileToDirectory(srcFile,destFile);// FileUtils工具类拷贝文件夹到指定文件夹File srcFile = new File("day17\\ddd");File destFile = new File("day17\\eee");FileUtils.copyDirectoryToDirectory(srcFile,destFile);}

总结

  • 能够使用字节缓冲流读取数据到程序
    使用InputStream流中的read() read(byte[] bys)
  • 能够使用字节缓冲流写出数据到文件
    使用OutputStream流中的write(int bys) write(byte[] bys,int off,int len)
  • 能够明确字符缓冲流的作用和基本用法
    以字符为单位读写数据,读数据,和写数据 (操作单位是字符)
  • 能够使用缓冲流的特殊功能
    BufferedReader:读取一行 readLine()
    BuffereWriter:写换行 newLine()
  • 能够阐述编码表的意义
    一套字符与二进制数之间的对应规则
  • 能够使用转换流读取指定编码的文本文件
    InputStreamReader(InputStream is,String charset)
  • 能够使用转换流写入指定编码的文本文件
    OutputStreamWriter(OutputStream is,String charset)
  • 能够使用序列化流写出对象到文件
    ObjectOutputStream: writeObject()
  • 能够使用反序列化流读取文件到程序中
    ObjectInputStream: readObject()
  • 能够理解装饰模式的实现步骤
    装饰类和被装饰类实现相同的接口
    在装饰类中需要获取被装饰类的引用
    在装饰类对不需要增强的方法,就使用被装饰类的引用调用被装饰类的方法
    在装饰类中对需要增强的方法进行增强
  • 能够使用commons-io工具包
    下载工具包
    把工具包拷贝到模块
    把工具包添加classpath路径

练习模块

题目一

请编写main()方法,定义一个存储String的集合:List,并初始化以下数据:

​ List list = new ArrayList<>();

​ list.add(“迪丽热巴”);

​ list.add(“古力娜扎”);

​ list.add(“周杰伦”);

​ list.add(“蔡徐坤”);

​ //请定义“字符缓冲输出流”BufferedWriter将集合中的数据写入到文件:test3_2.txt中,每个名字一行。

package com.day17;import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;public class Task1_2 {public static void main(String[] args) throws IOException {List<String> list = new ArrayList<>();list.add("迪丽热巴");list.add("古力娜扎");list.add("周杰伦");list.add("蔡徐坤");BufferedWriter br = new BufferedWriter(new FileWriter("Task1_2.txt"));for (String str : list) {br.write(str);br.newLine();br.flush();}br.close();}
}

第二题

编写main()方法,定义一个字符缓冲输入流BufferedReader,读取test3_2.txt文件,一次读取一行,将读取的内容存储到一个List集合中,遍历、并打印集合中的每个元素:

​ //1.定义一个字符缓冲输入流

​ BufferedReader in = new BufferedReader(…);

​ //2.定义一个集合

​ List list = new ArrayList<>();

​ //3.一次读取一行

​ …

​ …

package com.day17;import java.io.BufferedReader;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.List;public class Task1_3 {public static void main(String[] args) throws Exception{BufferedReader br = new BufferedReader(new FileReader("Task1_2.txt"));List<String> list = new ArrayList<>();String str;while ((str = br.readLine())!=null){list.add(str);}br.close();for (String s : list) {System.out.println(s);}}
}

第三题

请按以下要求编写程序:

​ 1).定义一个学员类,有以下属性:姓名、性别、年龄、分数

​ 无参、全参构造方法,get/set方法

​ 2).定义main()方法,定义一个存储Student的集合,并初始化一些数据:

​ List stuList = new ArrayList<>();

​ stuList.add(new Student(“迪丽热巴”,”女”,18,99);

​ stuList.add(new Student(“古力娜扎”,”女”,19,98);

​ stuList.add(new Student(“周杰伦”,”男”,20,88);

​ stuList.add(new Student(“蔡徐坤”,”男”,19,78);

​ /*

​ 定义一个字符缓冲输出流BufferedWriter,将学员信息写入到”test3_4.txt”中,

​ 每个学员信息占一行,每个字段之间用,符号隔开,例如:

​ 迪丽热巴,女,18,99

​ 古力娜扎,女,19,98

​ 周杰伦,男,20,88

​ 蔡徐坤,男,19,78

package com.day17;import java.io.BufferedWriter;
import java.io.FileWriter;
import java.io.IOException;
import java.util.ArrayList;
import java.util.List;public class Task1_4 {public static void main(String[] args) throws IOException {List<Student> stuList = new ArrayList<>();stuList.add(new Student("迪丽热巴","女",18,99));stuList.add(new Student("古力娜扎","女",19,98));stuList.add(new Student("周杰伦","男",20,88));stuList.add(new Student("蔡徐坤","男",19,78));BufferedWriter br = new BufferedWriter(new FileWriter("Task17_1_4.txt"));for (Student student : stuList) {String str = student.getName()+","+student.getGender()+","+student.getAge()+","+student.getScore();br.write(str);br.flush();br.newLine();}br.close();}
}
class Student{private String name;private String gender;private int age;private int score;public Student() {}public Student(String name, String gender, int age, int score) {this.name = name;this.gender = gender;this.age = age;this.score = score;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getGender() {return gender;}public void setGender(String gender) {this.gender = gender;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public int getScore() {return score;}public void setScore(int score) {this.score = score;}
}

第四题

​ 请编写main()方法,定义一个字符缓冲输入流BufferedReader对象,读取”test3_4.txt”文件,一次读取一行,将每行数据封装为一个Student对象,并将Student对象存储到一个集合。遍历并打印集合的所有Student信息。

​ //定义一个BufferedReader对象

​ BufferedReader in = new BufferedReader(…);

​ //定义一个集合

​ List stuList = new ArrayList<>();

​ //一次读取一行

​ …

package com.day17;import java.io.BufferedReader;
import java.io.FileReader;
import java.util.ArrayList;
import java.util.List;public class Task1_5 {public static void main(String[] args) throws Exception {//定义一个BufferedReader对象BufferedReader br = new BufferedReader(new FileReader("Task17_1_4.txt"));//定义一个集合List<Student> list = new ArrayList<Student>();String line ;while ((line = br.readLine())!=null){String [] str = line.split(",");list.add(new Student(str[0],str[1],Integer.valueOf(str[2]),Integer.valueOf(str[3])));}for (Student student : list) {System.out.println(student.getName()+","+student.getGender()+","+student.getAge()+","+student.getScore());}}
}class Student{private String name;private String gender;private int age;private int score;public Student() {}public Student(String name, String gender, int age, int score) {this.name = name;this.gender = gender;this.age = age;this.score = score;}public String getName() {return name;}public void setName(String name) {this.name = name;}public String getGender() {return gender;}public void setGender(String gender) {this.gender = gender;}public int getAge() {return age;}public void setAge(int age) {this.age = age;}public int getScore() {return score;}public void setScore(int score) {this.score = score;}
}

day17【缓冲流、转换流、序列化流、装饰者模式、commons-io工具包】相关推荐

  1. 集合到文件 文件到集合 点名器 集合到文件数据排序版 标准输入流 打印流 对象序列化流 对象返序列化流 Properties 游戏次数 进程和线程 线程 多线程的实现方式 设置和获取线程名称

    文章目录 集合到文件 文件到集合 点名器 集合到文件数据排序版 标准输入流 打印流 对象序列化流 对象返序列化流 Properties 游戏次数 进程和线程 线程 多线程的实现方式 设置和获取线程名称 ...

  2. Java基础-22总结登录注册IO版,数据操作流,内存操作流,打印流,标准输入输出流,转换流,随机访问流,合并流,序列化流,Properties...

    你需要的是什么,直接评论留言. 获取更多资源加微信公众号"Java帮帮" (是公众号,不是微信好友哦) 还有"Java帮帮"今日头条号,技术文章与新闻,每日更新 ...

  3. Java基础-22总结登录注册IO版,数据操作流,内存操作流,打印流,标准输入输出流,转换流,随机访问流,合并流,序列化流,Properties

    你需要的是什么,直接评论留言. 获取更多资源加微信公众号"Java帮帮" (是公众号,不是微信好友哦) 还有"Java帮帮"今日头条号,技术文章与新闻,每日更新 ...

  4. File类,字节字符输入输出流,缓冲流,标准流,对象序列化流

    一,File文件类 1 File类创建功能:    public boolean createNewFiLe():当具有该名称的文件不存在时,创建一个由该抽象路径名命名的新空文件       如果文件 ...

  5. java-IO流(5)-IO流中的设计模式(装饰器模式和适配器模式)的介绍

    目录 1装饰器模式 1.1定义 1.2代码实现 1.3装饰器特点 1.4装饰器在IO流中的使用 2配适器模式 2.1Adapter适配器 2.2代码实例 2.3适配器特点 2.4适配器优缺点 2.5适 ...

  6. IO流案例以及特殊操作流/对象对象序列化流/Properties集合

    文章目录 IO流案例 复制单级文件夹 复制多级文件夹 IO特殊操作流 标准输入流 字节打印流 字符打印流 对象序列化流 serialVersionUID&transient Propertie ...

  7. java装饰模式模拟流_Java设计模式--装饰器模式到Java IO 流

    装饰器模式 抽象构件角色:给出一个抽象接口,以规范准备接受附加责任的对象. 具体构件角色:定义准备接受附加责任的对象. 抽象装饰角色:持有一个构件对象的实例,并对应一个与抽象构件接口一致的接口. 具体 ...

  8. 【小白学Java】D32》》》IO流 之 序列化流 打印流

    [友情链接]➡➡▶IO流 之 File类 & 递归 [友情链接]➡➡▶IO流 之 过滤器 & 字节流 [友情链接]➡➡▶IO流 之字符流 & 属性集(Properties集合) ...

  9. Java19-day10【标准输入输出流、字节字符打印流、对象序列化-反序列化流、serialVersionUIDtransient、Properties】

    视频+资料[链接:https://pan.baidu.com/s/1MdFNUADVSFf-lVw3SJRvtg   提取码:zjxs] Java基础--学习笔记(零起点打开java世界的大门)--博 ...

最新文章

  1. 【二级java】软件工程基础
  2. Windows Live Messenger 8.5 抢先试用
  3. python编程django项目django.template.exceptions.TemplateDoesNotExist: registration/login.html解决方法
  4. 纷享销客完成新一轮数亿元融资,持续领跑中国CRM产业发展
  5. netbeans字体与颜色配置模板相关网站
  6. 小鹏汽车北京车展发布免费加电、电池租赁计划以及低空飞行汽车
  7. A Software Developer’s Reading Plan
  8. Mac OS X 10.8.3反编译Android apk
  9. SQL递归查询(with cte as)
  10. Pycharm 主题设置和修改
  11. 上市公司创新研发支出数据(2006-2018年)
  12. Vue设置浏览器小图标(ICON)
  13. 分部积分出现积回去的情况
  14. 音频交流会(个人学习向项目)
  15. “一对一直播软件”的开发,离不开直播源码开发+系统搭建
  16. 一个冷门json类JacksonJsonParser中的坑
  17. 他们十年发生的那些事,你好奇不?
  18. java多线程高级:JUC
  19. 最全的ORACLE-SQL笔记(转,出处不详)
  20. Qt-Onvif客户端

热门文章

  1. OpenGL的画笔工具GL10
  2. 给即将大三找方向的同学:放下焦虑,行动起来
  3. android上传图片并附带上传数据,文件流
  4. 自动驾驶涉及的核心技术与能力(持续更新中)
  5. 关于Eclipse中XML、JSP、JS等文件打开方式的设置
  6. 开智用c语言,在开智领悟到「一切技能皆可习得」后,我的人生轨迹就此改写...
  7. Animation Introduce(动画介绍) —— 一拳超人第一季
  8. 新书推荐|《扬帆远航:5G融合应用实践精编》(赠书福利)
  9. 接路由器后某些网页打不开的解决方法
  10. linux内核版本 2.6.14,Linux-2.6.14内核在S3C2410上的移植