1.File类

1.1 File类概述和构造方法【应用】

  • File类介绍
    • 它是文件和目录路径名的抽象表示
    • 文件和目录是可以通过File封装成对象的
    • 对于File而言,其封装的并不是一个真正存在的文件,仅仅是一个路径名而已.它可以是存在的,也可以是不存在的.将来是要通过具体的操作把这个路径的内容转换为具体存在的
  • File类的构造方法 | 方法名 | 说明 | | —- | —- | | File(String pathname) | 根据字符串路径创建一个File对象 | | File(String parent, String child) | 根据字符串父级路径和子级路径创建File对象 | | File(File parent, String child) | 根据File类型父级路径和字符串子级路径创建File对象 |

  • 示例代码

    1. public class FileDemo01 {
    2. public static void main(String[] args) {
    3. //通过将给定的路径名字符串转换为抽象路径名来创建新的 File实例
    4. File f1 = new File("E:\\itcast\\java.txt");
    5. System.out.println(f1);
    6. //从父路径名字符串和子路径名字符串创建新的 File实例
    7. File f2 = new File("E:\\itcast","java.txt");
    8. System.out.println(f2);
    9. //从父抽象路径名和子路径名字符串创建新的 File实例
    10. File f3 = new File("E:\\itcast");
    11. File f4 = new File(f3,"java.txt");
    12. System.out.println(f4);
    13. }
    14. }

1.2 绝对路径和相对路径【理解】

  • 绝对路径
    是一个完整的路径,从盘符开始
  • 相对路径
    是一个简化的路径,相对当前项目下的路径
  • 示例代码

    1. public class FileDemo02 {
    2. public static void main(String[] args) {
    3. // 是一个完整的路径,从盘符开始
    4. File file1 = new File("D:\\itheima\\a.txt");
    5. // 是一个简化的路径,从当前项目根目录开始
    6. File file2 = new File("a.txt");
    7. File file3 = new File("模块名\\a.txt");
    8. }
    9. }
    1. //绝对路径
    2. File file = new File("E:\\idea_project\\javase_advance\\day_11\\file\\demo01.txt");
    3. System.out.println(file.getAbsolutePath());
    4. //相对路径
    5. File file1 = new File("day_11\\file\\demo01.txt");
    6. System.out.println(file1.getAbsolutePath());
    7. //磁盘根目录
    8. File file2 = new File("\\day_11\\file\\demo01.txt");
    9. System.out.println(file2.getAbsolutePath());

1.3 File类创建功能【应用】

  • 方法分类 | 方法名 | 说明 | | —- | —- | | public boolean createNewFile() | 创建一个新空文件 | | public boolean mkdir() | 创建单级文件夹 | | public boolean mkdirs() | 创建多级文件夹 |

  • 示例代码 ```java public class FileDemo02 { public static void main(String[] args) throws IOException {

    1. //method1();
    2. //method2();
    3. //method2();

    }

    //1.如果文件存在,那么创建失败,返回false //2.如果文件不存在,那么创建成功,返回true //3.createNewFile方法不管调用者有没有后缀名,只能创建文件. private static void method1() throws IOException {

    1. File file1 = new File("C:\\itheima\\aaa");
    2. boolean result1 = file1.createNewFile();
    3. System.out.println(result1);

    }

    //1.只能创建单级文件夹,不能创建多级文件夹 //2.不管调用者有没有后缀名,只能创建单级文件夹 private static void method2() {

    1. File file = new File("C:\\itheima\\aaa");
    2. boolean result = file.mkdir();
    3. System.out.println(result);

    }

    //1.可以创建单级文件夹,也可以创建多级文件夹 //2.不管调用者有没有后缀名,只能创建单级文件夹 private static void method3() {

    1. File file = new File("C:\\itheima\\aaa\\bbb\\ccc");
    2. boolean result = file.mkdirs();
    3. System.out.println(result);

}

  1. <a name="4b370d0d"></a>
  2. ### 1.4 File类删除功能【应用】
  3. - 方法分类
  4. | 方法名 | 说明 |
  5. | --- | --- |
  6. | public boolean delete() | 删除文件或文件夹 |
  7. - 删除方法注意事项:
  8. - delete方法直接删除不走回收站
  9. - 如果删除的是一个文件,直接删除
  10. - 如果删除的是一个文件夹,需要先删除文件夹中的内容,最后才能删除文件夹
  11. - 示例代码
  12. ```java
  13. public class FileDemo03 {
  14. public static void main(String[] args) {
  15. //注意点:
  16. //1.不走回收站的.
  17. //2.如果删除的是文件,那么直接删除.如果删除的是文件夹,那么能删除空文件夹
  18. //3.如果要删除一个有内容的文件夹,只能先进入到这个文件夹,把里面的内容全部删除完毕,才能再次删除这个文件夹
  19. //简单来说:
  20. //只能删除文件和空文件夹.
  21. public static void main(String[] args) {
  22. //method1();
  23. method2();
  24. }
  25. //删除文件
  26. private static void method1() {
  27. File file = new File("d:\\itheima\\a.txt");
  28. boolean result = file.delete();
  29. System.out.println(result);
  30. }
  31. //删除非空目录
  32. private static void method2() {
  33. File file = new File("d:\\itheima");
  34. boolean result = file.delete();
  35. System.out.println(result);
  36. }
  37. }
  38. }

1.5 File类判断和获取功能【应用】

  • 判断功能 | 方法名 | 说明 | | —- | —- | | public boolean isDirectory() | 测试此抽象路径名表示的File是否为目录 | | public boolean isFile() | 测试此抽象路径名表示的File是否为文件 | | public boolean exists() | 测试此抽象路径名表示的File是否存在 |

  • 获取功能 | 方法名 | 说明 | | —- | —- | | public String getAbsolutePath() | 返回此抽象路径名的绝对路径名字符串 | | public String getPath() | 将此抽象路径名转换为路径名字符串 | | public String getName() | 返回由此抽象路径名表示的文件或目录的名称 |

  • 示例代码

    1. /**
    2. * File的判断和获取功能
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws IOException {
    6. File f = new File("day_11\\lv.txt");
    7. //测试此抽象路径名表示的File是否为目录
    8. System.out.println(f.isDirectory());
    9. //测试此抽象路径名表示的File是否为文件
    10. System.out.println(f.isFile());
    11. //测试此抽象路径名表示的File是否存在
    12. System.out.println(f.exists());
    13. //返回此抽象路径名的绝对路径名字符串
    14. System.out.println(f.getAbsolutePath());
    15. //将此抽象路径名转换为路径名字符串
    16. System.out.println(f.getPath());
    17. //返回由此抽象路径名表示的文件或目录的名称
    18. System.out.println(f.getName());
    19. System.out.println("--------");
    20. }
    21. }

1.6 File的FileList方法【应用】

  • File[] listFiles(): 获取文件夹下的所有文件和文件夹对象,封装到File数组中返回
  • listFiles方法注意事项
    • 当调用者不存在时,返回null
    • 当调用者是一个文件时,返回null
    • 当调用者是一个空文件夹时,返回一个长度为0的数组
    • 当调用者是一个有内容的文件夹时,将里面所有文件和文件夹的路径放在File数组中返回
    • 当调用者是一个有隐藏文件的文件夹时,将里面所有文件和文件夹的路径放在File数组中返回,包含隐藏内容
    • 当调用者是一个需要权限才能进入的文件夹时,返回null
  • 示例代码
    1. /**
    2. * File的listFile方法
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) {
    6. File file = new File("D:\\Tools");
    7. File[] files = file.listFiles();
    8. for (File f : files) {
    9. System.out.println(f);
    10. }
    11. }
    12. }

1.7 File类练习一【应用】

  • 案例需求
    在当前模块下的aaa文件夹中创建一个a.txt文件
  • 实现步骤
    • 创建File对象,指向aaa文件夹
    • 判断aaa文件夹是否存在,如果不存在则创建
    • 创建File对象,指向aaa文件夹下的a.txt文件
    • 创建这个文件
  • 代码实现
    1. public class Test1 {
    2. public static void main(String[] args) throws IOException {
    3. //1.创建File对象,指向aaa文件夹
    4. File file = new File("filemodule\\aaa");
    5. //2.判断aaa文件夹是否存在,如果不存在则创建
    6. if(!file.exists()){
    7. //如果文件夹不存在,就创建出来
    8. file.mkdirs();
    9. }
    10. //3.创建File对象,指向aaa文件夹下的a.txt文件
    11. File newFile = new File(file,"a.txt");
    12. //4.创建这个文件
    13. newFile.createNewFile();
    14. }
    15. }

1.8 File类练习二【应用】

  • 案例需求
    删除一个多级文件夹
  • 实现步骤
    • 定义一个方法,接收一个File对象
    • 遍历这个File对象,获取它下边的每个文件和文件夹对象
    • 判断当前遍历到的File对象是文件还是文件夹
    • 如果是文件,直接删除
    • 如果是文件夹,递归调用自己,将当前遍历到的File对象当做参数传递
    • 参数传递过来的文件夹File对象已经处理完成,最后直接删除这个空文件夹
  • 代码实现

    1. public class Test2 {
    2. public static void main(String[] args) {
    3. //练习二:删除一个多级文件夹
    4. //delete方法
    5. //只能删除文件和空文件夹.
    6. //如果现在要删除一个有内容的文件夹?
    7. //先删掉这个文件夹里面所有的内容.
    8. //最后再删除这个文件夹
    9. File src = new File("C:\\Users\\apple\\Desktop\\src");
    10. deleteDir(src);
    11. }
    12. //1.定义一个方法,接收一个File对象
    13. private static void deleteDir(File src) {
    14. //先删掉这个文件夹里面所有的内容.
    15. //递归 方法在方法体中自己调用自己.
    16. //注意: 可以解决所有文件夹和递归相结合的题目
    17. //2.遍历这个File对象,获取它下边的每个文件和文件夹对象
    18. File[] files = src.listFiles();
    19. //3.判断当前遍历到的File对象是文件还是文件夹
    20. for (File file : files) {
    21. //4.如果是文件,直接删除
    22. if(file.isFile()){
    23. file.delete();
    24. }else{
    25. //5.如果是文件夹,递归调用自己,将当前遍历到的File对象当做参数传递
    26. deleteDir(file);//参数一定要是src文件夹里面的文件夹File对象
    27. }
    28. }
    29. //6.参数传递过来的文件夹File对象已经处理完成,最后直接删除这个空文件夹
    30. src.delete();
    31. }
    32. }
  • 思路解析

2020-07-18_153018.png

1.8 File类练习三【应用】

  • 案例需求
    统计一个文件夹中每种文件的个数并打印
    打印格式如下:

    1. txt:3
    2. doc:4
    3. jpg:6
  • 实现步骤

    • 定义一个方法,参数是HashMap集合用来统计次数和File对象要统计的文件夹
    • 遍历File对象,获取它下边的每一个文件和文件夹对象
    • 判断当前File对象是文件还是文件夹
    • 如果是文件,判断这种类型文件后缀名在HashMap集合中是否出现过
      • 没出现过,将这种类型文件的后缀名存入集合中,次数存1
      • 出现过,获取这种类型文件的后缀名出现的次数,对其+1,在存回集合中
    • 如果是文件夹,递归调用自己,HashMap集合就是参数集合,File对象是当前文件夹对象
  • 代码实现

    1. public class Test3 {
    2. public static void main(String[] args) {
    3. //统计一个文件夹中,每种文件出现的次数.
    4. //统计 --- 定义一个变量用来统计. ---- 弊端:同时只能统计一种文件
    5. //利用map集合进行数据统计,键 --- 文件后缀名 值 ---- 次数
    6. File file = new File("filemodule");
    7. HashMap<String, Integer> hm = new HashMap<>();
    8. getCount(hm, file);
    9. System.out.println(hm);
    10. }
    11. //1.定义一个方法,参数是HashMap集合用来统计次数和File对象要统计的文件夹
    12. private static void getCount(HashMap<String, Integer> hm, File file) {
    13. //2.遍历File对象,获取它下边的每一个文件和文件夹对象
    14. File[] files = file.listFiles();
    15. for (File f : files) {
    16. //3.判断当前File对象是文件还是文件夹
    17. if(f.isFile()){
    18. //如果是文件,判断这种类型文件后缀名在HashMap集合中是否出现过
    19. String fileName = f.getName();
    20. String[] fileNameArr = fileName.split("\\.");
    21. if(fileNameArr.length == 2){
    22. String fileEndName = fileNameArr[1];
    23. if(hm.containsKey(fileEndName)){
    24. //出现过,获取这种类型文件的后缀名出现的次数,对其+1,在存回集合中
    25. Integer count = hm.get(fileEndName);
    26. //这种文件又出现了一次.
    27. count++;
    28. //把已经出现的次数给覆盖掉.
    29. hm.put(fileEndName,count);
    30. }else{
    31. // 没出现过,将这种类型文件的后缀名存入集合中,次数存1
    32. hm.put(fileEndName,1);
    33. }
    34. }
    35. }else{
    36. //如果是文件夹,递归调用自己,HashMap集合就是参数集合,File对象是当前文件夹对象代码实现
    37. getCount(hm,f);
    38. }
    39. }
    40. }
    41. }

2.字节流

2.1 IO流概述和分类【理解】

  • IO流介绍
    • IO:输入/输出(Input/Output)
    • 流:是一种抽象概念,是对数据传输的总称.也就是说数据在设备间的传输称为流,流的本质是数据传输
    • IO流就是用来处理设备间数据传输问题的.常见的应用: 文件复制; 文件上传; 文件下载
  • IO流的分类
    • 按照数据的流向
      • 输入流:读数据
      • 输出流:写数据
    • 按照数据类型来分
      • 字节流
        • 字节输入流
        • 字节输出流
      • 字符流
        • 字符输入流
        • 字符输出流
  • IO流的使用场景
    • 如果操作的是纯文本文件,优先使用字符流
    • 如果操作的是图片、视频、音频等二进制文件,优先使用字节流
    • 如果不确定文件类型,优先使用字节流.字节流是万能的流

1018541-20170317090827791-1722217449.jpg

  • 这么庞大的体系里面,常用的就那么几个,我们把它们抽取出来,如下图:

1018541-20170317090935213-142491173.png

2.2 字节流写数据【应用】

  • 字节流抽象基类
    • InputStream:这个抽象类是表示字节输入流的所有类的超类
    • OutputStream:这个抽象类是表示字节输出流的所有类的超类
    • 子类名特点:子类名称都是以其父类名作为子类名的后缀
  • 字节输出流
    • FileOutputStream(String name):创建文件输出流以指定的名称写入文件
  • 使用字节输出流写数据的步骤
    1. 创建FileOutputStream对象,关联到一个文件路径
    2. 调用write()方法,写出数据
    3. 调用close()方法,释放资源
  • 示例代码

    1. public class FileOutputStreamDemo01 {
    2. public static void main(String[] args) throws IOException {
    3. //创建字节输出流对象
    4. /*
    5. 注意点:
    6. 1.如果文件不存在,会帮我们创建
    7. 2.如果文件存在,会把文件清空
    8. */
    9. //FileOutputStream(String name):创建文件输出流以指定的名称写入文件
    10. FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
    11. //void write(int b):将指定的字节写入此文件输出流
    12. fos.write(97);
    13. // fos.write(57);
    14. // fos.write(55);
    15. //最后都要释放资源
    16. //void close():关闭此文件输出流并释放与此流相关联的任何系统资源。
    17. fos.close();
    18. }
    19. }
  • 注意事项
    1.如果目的地指向的文件不存在,会自动帮我们创建
    2.如果存在,会先将文件清空,在进行写入数据
    3.通过IO流操作文件后,必须关流释放资源,否则文件会一直被占用

2.3 字节流一次写多个数据【应用】

  • 写数据的方法分类 | 方法名 | 说明 | | —- | —- | | void write(int b) | 将指定的字节写入此文件输出流 一次写一个字节数据 | | void write(byte[] b) | 将 b.length字节从指定的字节数组写入此文件输出流 一次写一个字节数组数据 | | void write(byte[] b, int off, int len) | 将 len字节从指定的字节数组开始,从偏移量off开始写入此文件输出流 一次写一个字节数组的部分数据 |

  • 示例代码

    1. public class FileOutputStreamDemo02 {
    2. public static void main(String[] args) throws IOException {
    3. FileOutputStream fos = new FileOutputStream("bytestream\\a.txt");
    4. /*byte [] bys = {97,98,99};
    5. fos.write(bys);*/
    6. byte [] bys = {97,98,99,100,101,102,103};
    7. fos.write(bys,1,2);
    8. fos.close();
    9. }

2.4 字节流写数据的两个小问题【应用】

  • 字节流写数据如何实现换行
    • windows:\r\n
    • linux:\n
    • mac:\r
  • 字节流写数据如何实现追加写入
    • public FileOutputStream(String name,boolean append)
    • 创建文件输出流以指定的名称写入文件。如果第二个参数为true ,则字节将写入文件的末尾而不是开头
  • 示例代码

    1. public class FileOutputStreamDemo03 {
    2. public static void main(String[] args) throws IOException {
    3. //创建字节输出流对象
    4. // FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt");
    5. FileOutputStream fos = new FileOutputStream("myByteStream\\fos.txt",true);
    6. //写数据
    7. for (int i = 0; i < 10; i++) {
    8. fos.write("hello".getBytes());
    9. fos.write("\r\n".getBytes());
    10. }
    11. //释放资源
    12. fos.close();
    13. }
    14. }

2.5 字节流写数据加异常处理【应用】

  • 异常处理格式

    • try-catch-finally

      1. try{
      2. 可能出现异常的代码;
      3. }catch(异常类名 变量名){
      4. 异常的处理代码;
      5. }finally{
      6. 执行所有清除操作;
      7. }
    • finally特点

      • 被finally控制的语句一定会执行,除非JVM退出
  • 示例代码
    1. public class FileOutputStreamDemo04 {
    2. public static void main(String[] args) {
    3. //加入finally来实现释放资源
    4. FileOutputStream fos = null;
    5. try {
    6. fos = new FileOutputStream("myByteStream\\fos.txt");
    7. fos.write("hello".getBytes());
    8. } catch (IOException e) {
    9. e.printStackTrace();
    10. } finally {
    11. if(fos != null) {
    12. try {
    13. fos.close();
    14. } catch (IOException e) {
    15. e.printStackTrace();
    16. }
    17. }
    18. }
    19. }
    20. }

2.6 字节输出流小结【应用】

  • 创建字节输出流对象,构造方法中传入目的地文件路径
    • 如果文件不存在,会自动帮我们创建
    • 如果文件存在,会先清空,在写入数据,如果不想被清空,第二个参数传true
  • 写出数据
    • 可以一次写一个字节,可以一次写一个字节数组,可以一次写一个字节数组的一部分
    • 写入换行: \r\n
  • 关闭流,释放资源

2.7 字节流读数据(一次读一个字节数据)【应用】

  • 字节输入流
    • FileInputStream(String name):通过打开与实际文件的连接来创建一个FileInputStream,该文件由文件系统中的路径名name命名
  • 字节输入流读取数据的步骤
    • 创建字节输入流对象
    • 调用字节输入流对象的读数据方法
    • 释放资源
  • 示例代码

    1. public class FileInputStreamDemo01 {
    2. public static void main(String[] args) throws IOException {
    3. //创建字节输入流对象
    4. FileInputStream fis = new FileInputStream("myByteStream\\fos.txt");
    5. int read = fis.read();
    6. System.out.println((char)read);
    7. fis.close();
    8. //释放资源
    9. fis.close();
    10. }
    11. }

2.8 字节流-读取多个字节【应用】

  • 读取文件中多个字节重要步骤

    1. 1.创建FileInputStream对象,关联文件
    2. 2.通过循环读取文件,调用read()方法
    3. 3.如果读取到的是-1,则停止循环
    4. 4.调用close()方法释放资源
  • 示例代码

    1. public class OutputDemo8 {
    2. public static void main(String[] args) throws IOException {
    3. FileInputStream fis = new FileInputStream("bytestream\\a.txt");
    4. //1,文件中多个字节我怎么办?
    5. /*while(true){
    6. int i1 = fis.read();
    7. System.out.println(i1);
    8. }*/
    9. int b;
    10. while ((b = fis.read())!=-1){
    11. System.out.println((char) b);
    12. }
    13. fis.close();
    14. }
    15. }

2.9 字节流复制文件【应用】

  • 案例需求
    把“E:\itcast\窗里窗外.txt”复制到模块目录下的“窗里窗外.txt” (文件可以是任意文件)
  • 实现步骤
    • 复制文本文件,其实就把文本文件的内容从一个文件中读取出来(数据源),然后写入到另一个文件中(目的地)
    • 数据源:
      E:\itcast\窗里窗外.txt —- 读数据 —- InputStream —- FileInputStream
    • 目的地:
      myByteStream\窗里窗外.txt —- 写数据 —- OutputStream —- FileOutputStream
  • 代码实现

    1. public class CopyTxtDemo {
    2. public static void main(String[] args) throws IOException {
    3. //根据数据源创建字节输入流对象
    4. FileInputStream fis = new FileInputStream("E:\\itcast\\窗里窗外.txt");
    5. //根据目的地创建字节输出流对象
    6. FileOutputStream fos = new FileOutputStream("myByteStream\\窗里窗外.txt");
    7. //读写数据,复制文本文件(一次读取一个字节,一次写入一个字节)
    8. int by;
    9. while ((by=fis.read())!=-1) {
    10. fos.write(by);
    11. }
    12. //释放资源
    13. fos.close();
    14. fis.close();
    15. }
    16. }

2.10 字节流-文件复制(一次读一个字节数组数据)【应用】

  • 一次读一个字节数组的方法
    • public int read(byte[] b):从输入流读取最多b.length个字节的数据
    • 返回的是读入缓冲区的总字节数,也就是实际读取字节个数
  • 示例代码

    1. public class CopyJpgDemo {
    2. public static void main(String[] args) throws IOException {
    3. //根据数据源创建字节输入流对象
    4. FileInputStream fis = new FileInputStream("E:\\itcast\\mn.jpg");
    5. //根据目的地创建字节输出流对象
    6. FileOutputStream fos = new FileOutputStream("myByteStream\\mn.jpg");
    7. //读写数据,复制图片(一次读取一个字节数组,一次写入一个字节数组)
    8. byte[] bys = new byte[1024];
    9. int len;//实际读取字节个数
    10. while ((len=fis.read(bys))!=-1) {
    11. fos.write(bys,0,len);
    12. }
    13. //释放资源
    14. fos.close();
    15. fis.close();
    16. }
    17. }

2.11 小数组拷贝原理【理解】

注意:变量len接收的返回值不是数组的长度,而是数组中真正存储数据的个数
2020-07-18_175458.png

3.字节缓冲流

3.1 字节缓冲流构造方法【应用】

  • 字节缓冲流:

    • BufferedOutputStream: 字节缓冲输出流
    • BufferedInputStream: 字节缓冲输入流
  • 构造方法:

    • 字节缓冲输出流: BufferedOutputStream(OutputStream out)
    • 字节缓冲输入流: BufferedInputStream(InputStream in)
  • 读和写的方法:

    • BufferedOutputStream: 从父类OutputStream继承写的方法
    • BufferedInputStream: 从父类InputStream继承读的方法
  • 示例代码

    1. /**
    2. * 缓冲流-一次读写一个字节
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws IOException {
    6. //就要利用缓冲流去拷贝文件
    7. //创建一个字节缓冲输入流
    8. //在底层创建了一个默认长度为8192的字节数组。
    9. BufferedInputStream bis = new BufferedInputStream(new FileInputStream("day_11\\file\\test.avi"));
    10. //创建一个字节缓冲输出流
    11. //在底层也创建了一个默认长度为8192的字节数组。
    12. BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("day_11\\file\\copyBuf.avi"));
    13. int b;
    14. while ((b = bis.read()) != -1) {
    15. bos.write(b);
    16. }
    17. bis.close();
    18. bos.close();
    19. }
    20. }

3.2 缓冲流—-一次读写一个字节原理【理解】

  • 缓冲流底层如何实现缓冲区的
    • 通过定义一个长度为8192的数组来实现缓冲区的

2020-07-18_191903.png

3.3 缓冲流—-一次读写一个字节数组【应用】

  • 原理2020-07-18_192515.png
  • 示例代码

    1. public class MainClass {
    2. public static void main(String[] args) throws IOException {
    3. //创建一个字节缓冲输入流
    4. //在底层创建了一个默认长度为8192的字节数组。
    5. BufferedInputStream bis = new BufferedInputStream(new FileInputStream("day_11\\file\\test.avi"));
    6. //创建一个字节缓冲输出流
    7. //在底层也创建了一个默认长度为8192的字节数组。
    8. BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("day_11\\file\\copyBuf-2.avi"));
    9. byte[] bytes = new byte[1024];
    10. int len;
    11. while ((len = bis.read(bytes)) != -1) {
    12. bos.write(bytes,0,len);
    13. }
    14. bis.close();
    15. bos.close();
    16. System.out.println("复制完成");
    17. }
    18. }

3.3 字节流-小结

  • 字节流
    • 可以操作(拷贝)所有类型的文件
  • 字节缓冲流
    • 可以提高效率
    • 不能直接操作文件,需要传递字节流
  • 拷贝文件的四种方式
    • 字节流一次读写一个字节
    • 字节流一次读写一个字节数组
    • 字节缓冲流一次读写一个字节
    • 字节缓冲流一次读写一个字节数组

4.字符流

4.1 为什么会出现字符流【理解】

  • 字符流的介绍
    • 由于字节流操作非英语语言不是特别的方便,所以Java就提供字符流
    • 字符流 = 字节流 + 编码表

4.2 编码表【理解】

  • 什么是编码?
    • 字符—————-编码表————————二进制
  • 什么是解码?
    • 二进制————编码表———————-字符
  • 用什么编码,就要用什么解码,否则会出现乱码

  • 什么是字符集

    • 是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
    • 计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码。常见字符集有ASCII字符集、GBXXX字符集、Unicode字符集等
  • 常见的字符集
    • ASCII字符集:
      • ASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
      • 基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
    • GBXXX字符集:
      • GBK,最常用的中文码表。是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等
    • Unicode字符集:
      • UTF-8编码:可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用 中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码。它使用一至四个字节为每个字符编码
      • 编码规则:
        • 128个US-ASCII字符,只需一个字节编码
        • 拉丁文等字符,需要二个字节编码
        • 大部分常用字(含中文),使用三个字节编码
        • 其他极少使用的Unicode辅助字符,使用四字节编码

2021-01-09_232701.png

4.3 字符串中的编码解码问题【应用】

  • 相关方法 | 方法名 | 说明 | | —- | —- | | byte[] getBytes() | 使用平台的默认字符集将该 String编码为一系列字节 | | byte[] getBytes(String charsetName) | 使用指定的字符集将该 String编码为一系列字节 | | String(byte[] bytes) | 使用平台的默认字符集解码指定的字节数组来创建字符串 | | String(byte[] bytes, String charsetName) | 通过指定的字符集解码指定的字节数组来创建字符串 |

  • 代码演示

    1. /**
    2. * 字符串的编码和解码
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws UnsupportedEncodingException {
    6. String str = "中国";
    7. //使用平台的默认字符集将该 String编码为一系列字节,将结果存储到新的字节数组中
    8. byte[] bytes1 = str.getBytes();
    9. System.out.println(Arrays.toString(bytes1));
    10. //使用指定的字符集将该 String编码为一系列字节,将结果存储到新的字节数组中
    11. byte[] bytes2 = str.getBytes("UTF-8");
    12. System.out.println(Arrays.toString(bytes2));
    13. //使用指定的字符集将该 String编码为一系列字节,将结果存储到新的字节数组中
    14. byte[] bytes3 = str.getBytes("GBK");
    15. System.out.println(Arrays.toString(bytes3));
    16. //利用默认的UTF-8进行解码
    17. String s1 = new String(bytes1);
    18. System.out.println(s1);
    19. //利用指定的UTF-8进行解码
    20. String s2 = new String(bytes2,"UTF-8");
    21. System.out.println(s2);
    22. //利用指定的GBK进行解码
    23. String s3 = new String(bytes3,"GBK");
    24. System.out.println(s3);
    25. }
    26. }

4.4 字节流读取中文出现乱码的原因

  • 每次读取一个字节,只是中文的一部分,不是完整的中文,所以会乱码2020-07-18_200925.png

4.5 字符流

  • 字符流 = 字节流 + 编码表
  • 汉字在存储的时候,无论选择哪种编码存储,第一个字节都是负数2020-07-18_201316.png
  • 结论:
    • 想要进行拷贝,一律使用字节流或者字节缓冲流
    • 想把文本文件中的数据读到内存中,请使用字符输入流。
    • 想把内存中的数据写到文本文件中,请使用字符输出流。
    • GBK码表一个中文两个字节,UTF-8编码格式一个中文3个字节

4.6 字符流写数据【应用】

1018541-20170317090935213-142491173.png

  • 介绍
    Writer: 用于写入字符流的抽象父类
    FileWriter: 用于写入字符流的常用子类
  • 构造方法 | 方法名 | 说明 | | —- | —- | | FileWriter(File file) | 根据给定的 File 对象构造一个 FileWriter 对象 | | FileWriter(File file, boolean append) | 根据给定的 File 对象构造一个 FileWriter 对象 | | FileWriter(String fileName) | 根据给定的文件名构造一个 FileWriter 对象 | | FileWriter(String fileName, boolean append) | 根据给定的文件名以及指示是否附加写入数据的 boolean 值来构造 FileWriter 对象 |

  • 成员方法 | 方法名 | 说明 | | —- | —- | | void write(int c) | 写一个字符 | | void write(char[] cbuf) | 写入一个字符数组 | | void write(char[] cbuf, int off, int len) | 写入字符数组的一部分 | | void write(String str) | 写一个字符串 | | void write(String str, int off, int len) | 写一个字符串的一部分 |

  • 代码演示

    1. /**
    2. * 字符流-写方法
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws IOException {
    6. Writer writer = new FileWriter("day_12\\file\\demo02.txt");
    7. //写一个字符
    8. writer.write('a');
    9. writer.write(97);
    10. writer.write("\r\n");
    11. //写出一个字符数组
    12. char [] chars1 = {97,98,99,100,101};
    13. writer.write(chars1);
    14. writer.write("\r\n");
    15. //写出字符数组的一部分
    16. char [] chars2 = {97,98,99,100,101};
    17. writer.write(chars2,0,2);
    18. writer.write("\r\n");
    19. //写一个字符串
    20. writer.write("你好");
    21. writer.write("\r\n");
    22. //写一个字符串的一部分
    23. writer.write("你好",0,1);
    24. writer.close();
    25. System.out.println("end");
    26. }
    27. }

4.7 字符流写出数据注意事项

  • 如果文件不存在,就创建。但要保证父级路径存在
  • 如果文件存在会将内容清空
  • 写出int类型的整数,实际写出的是整数在码表上对应的字母
  • 有写出字符串的方法,可以直接写出字符串

4.8 刷新和关闭的方法

方法名 说明
flush() 刷新流,之后还可以继续写数据
close() 关闭流,释放资源,但是在关闭之前会先刷新流。一旦关闭,就不能再写数据

4.9 字符流读数据【应用】

  • 介绍
    Reader: 用于读取字符流的抽象父类
    FileReader: 用于读取字符流的常用子类
  • 构造方法 | 方法名 | 说明 | | —- | —- | | FileReader(File file) | 在给定从中读取数据的 File 的情况下创建一个新 FileReader | | FileReader(String fileName) | 在给定从中读取数据的文件名的情况下创建一个新 FileReader |
  • 成员方法 | 方法名 | 说明 | | —- | —- | | int read() | 一次读一个字符数据 | | int read(char[] cbuf) | 一次读一个字符数组数据 |

  • 代码演示

    1. public class InputStreamReaderDemo {
    2. public static void main(String[] args) throws IOException {
    3. FileReader fr = new FileReader("myCharStream\\b.txt");
    4. //int read():一次读一个字符数据
    5. // int ch;
    6. // while ((ch=fr.read())!=-1) {
    7. // System.out.print((char)ch);
    8. // }
    9. //int read(char[] cbuf):一次读一个字符数组数据
    10. char[] chs = new char[1024];
    11. int len;
    12. while ((len = fr.read(chs)) != -1) {
    13. System.out.print(new String(chs, 0, len));
    14. }
    15. //释放资源
    16. fr.close();
    17. }
    18. }

4.10 字符流用户注册案例【应用】

  • 案例需求
    将键盘录入的用户名和密码保存到本地实现永久化存储
  • 实现步骤
    • 获取用户输入的用户名和密码
    • 将用户输入的用户名和密码写入到本地文件中
    • 关流,释放资源
  • 代码实现

    1. public class CharStreamDemo8 {
    2. public static void main(String[] args) throws IOException {
    3. //需求: 将键盘录入的用户名和密码保存到本地实现永久化存储
    4. //要求:用户名独占一行,密码独占一行
    5. //分析:
    6. //1,实现键盘录入,把用户名和密码录入进来
    7. Scanner sc = new Scanner(System.in);
    8. System.out.println("请录入用户名");
    9. String username = sc.next();
    10. System.out.println("请录入密码");
    11. String password = sc.next();
    12. //2.分别把用户名和密码写到本地文件。
    13. FileWriter fw = new FileWriter("charstream\\a.txt");
    14. //将用户名和密码写到文件中
    15. fw.write(username);
    16. //表示写出一个回车换行符 windows \r\n MacOS \r Linux \n
    17. fw.write("\r\n");
    18. fw.write(password);
    19. //刷新流
    20. fw.flush();
    21. //3.关流,释放资源
    22. fw.close();
    23. }
    24. }

4.11 字符缓冲输入流【应用】

  • 字符缓冲流介绍
    • BufferedReader:从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以指定缓冲区大小,或者可以使用默认大小。 默认值足够大,可用于大多数用途。
    • 构造方法:
      • BufferedReader(Reader in)
      • BufferedReader(Reader in, int sz)
    • 成员方法:
      • 继承父类的成员方法进行使用
  • 代码演示

    1. /**
    2. * 字符缓冲输入流
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws IOException {
    6. //字符缓冲输入流
    7. BufferedReader br = new BufferedReader(new FileReader("day_12\\file\\demo06.txt"));
    8. //读取数据
    9. char [] chars = new char[1024];
    10. int len;
    11. while((len = br.read(chars)) != -1){
    12. System.out.println(new String(chars,0,len));
    13. }
    14. br.close();
    15. }
    16. }

1.12 字符缓冲输出流【应用】

  • 字符缓冲输出流的使用
    • BufferedWriter:将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大,可用于大多数用途
  • 构造方法:
    • BufferedWriter(Writer out)
    • BufferedWriter (Writer out, int sz)
  • 成员方法:
    • 继承父类的成员方法进行使用
  • 代码演示

    1. /**
    2. * 字符缓冲输出流
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws IOException {
    6. BufferedWriter bw = new BufferedWriter(new FileWriter("day_12\\file\\demo07.txt"));
    7. //写出数据
    8. //实际写出的是97对应的字符a
    9. bw.write(97);
    10. bw.write("\r\n");
    11. //实际写出的是97 - 101 对应的字符 abcde
    12. char [] chars = {97,98,99,100,101};
    13. bw.write(chars);
    14. bw.write("\r\n");
    15. //实际写的是abc
    16. bw.write(chars,0,3);
    17. bw.write("\r\n");
    18. //会把字符串的内容原样写出
    19. bw.write("黑马程序员");
    20. bw.write("\r\n");
    21. //会把字符串的一部分写出 abcde
    22. String line = "abcdefg";
    23. bw.write(line,0,5);
    24. bw.flush();
    25. bw.close();
    26. }
    27. }

4.13 字符缓冲流特有功能【应用】

  • 方法介绍
    BufferedWriter: | 方法名 | 说明 | | —- | —- | | void newLine() | 写一行行分隔符,行分隔符字符串由系统属性定义 |

  • BufferedReader: | 方法名 | 说明 | | —- | —- | | String readLine() | 读一行文字。 结果包含行的内容的字符串,不包括任何行终止字符如果流的结尾已经到达,则为null |

  • 代码演示

    1. public class BufferedStreamDemo02 {
    2. public static void main(String[] args) throws IOException {
    3. //创建字符缓冲输出流
    4. BufferedWriter bw = new BufferedWriter(new FileWriter("myCharStream\\bw.txt"));
    5. //写数据
    6. for (int i = 0; i < 10; i++) {
    7. bw.write("hello" + i);
    8. //bw.write("\r\n");
    9. bw.newLine();
    10. bw.flush();
    11. }
    12. //释放资源
    13. bw.close();
    14. //创建字符缓冲输入流
    15. BufferedReader br = new BufferedReader(new FileReader("myCharStream\\bw.txt"));
    16. String line;
    17. while ((line=br.readLine())!=null) {
    18. System.out.println(line);
    19. }
    20. br.close();
    21. }
    22. }

4.14 字符缓冲流操作文件中数据排序案例【应用】

  • 案例需求
    使用字符缓冲流读取文件中的数据,排序后再次写到本地文件
  • 实现步骤
    • 将文件中的数据读取到程序中
    • 对读取到的数据进行处理
    • 将处理后的数据添加到集合中
    • 对集合中的数据进行排序
    • 将排序后的集合中的数据写入到文件中
  • 代码实现

    1. public class CharStreamDemo14 {
    2. public static void main(String[] args) throws IOException {
    3. //需求:读取文件中的数据,排序后再次写到本地文件
    4. //分析:
    5. //1.要把文件中的数据读取进来。
    6. BufferedReader br = new BufferedReader(new FileReader("charstream\\sort.txt"));
    7. //输出流一定不能写在这里,因为会清空文件中的内容
    8. //BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));
    9. String line = br.readLine();
    10. System.out.println("读取到的数据为" + line);
    11. br.close();
    12. //2.按照空格进行切割
    13. String[] split = line.split(" ");//9 1 2 5 3 10 4 6 7 8
    14. //3.把字符串类型的数组变成int类型
    15. int [] arr = new int[split.length];
    16. //遍历split数组,可以进行类型转换。
    17. for (int i = 0; i < split.length; i++) {
    18. String smallStr = split[i];
    19. //类型转换
    20. int number = Integer.parseInt(smallStr);
    21. //把转换后的结果存入到arr中
    22. arr[i] = number;
    23. }
    24. //4.排序
    25. Arrays.sort(arr);
    26. System.out.println(Arrays.toString(arr));
    27. //5.把排序之后结果写回到本地 1 2 3 4...
    28. BufferedWriter bw = new BufferedWriter(new FileWriter("charstream\\sort.txt"));
    29. //写出
    30. for (int i = 0; i < arr.length; i++) {
    31. bw.write(arr[i] + " ");
    32. bw.flush();
    33. }
    34. //释放资源
    35. bw.close();
    36. }
    37. }

4.15 IO流小结【理解】

  • IO流小结01_IO流小结.png

5.转换流

5.1 字符流中和编码解码问题相关的两个类【理解】

  • InputStreamReader:是从字节流到字符流的桥梁,父类是Reader
    它读取字节,并使用指定的编码将其解码为字符
    它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集
  • OutputStreamWriter:是从字符流到字节流的桥梁,父类是Writer
    是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节
    它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台的默认字符集

5.2 转换流读写数据【应用】

  • 构造方法 | 方法名 | 说明 | | —- | —- | | InputStreamReader(InputStream in) | 使用默认字符编码创建InputStreamReader对象 | | InputStreamReader(InputStream in,String chatset) | 使用指定的字符编码创建InputStreamReader对象 | | OutputStreamWriter(OutputStream out) | 使用默认字符编码创建OutputStreamWriter对象 | | OutputStreamWriter(OutputStream out,String charset) | 使用指定的字符编码创建OutputStreamWriter对象 |

  • 代码演示-转换流

    1. /**
    2. * 转换输入流,字节转换为字符
    3. * 转换输出流,字符转换为字节
    4. */
    5. public class MainClass1 {
    6. public static void main(String[] args) throws IOException {
    7. //发生了乱码,因为demo10.txt文件的编码是GBK,IDEA中程序处理默认是UTF-8
    8. /*FileReader fr = new FileReader("day_12\\file\\demo10.txt");
    9. int ch;
    10. while ((ch = fr.read())!=-1){
    11. System.out.println((char)ch);
    12. }
    13. fr.close();*/
    14. //采用转换流指定编码格式
    15. /*InputStreamReader isr = new InputStreamReader(new FileInputStream("day_12\\file\\demo10.txt"),"GBK");
    16. int ch;
    17. while ((ch = isr.read())!=-1){
    18. System.out.print((char)ch);
    19. }
    20. isr.close();*/
    21. //JDK11后FileReader,支持指定编码格式
    22. FileReader fr = new FileReader("day_12\\file\\demo10.txt",Charset.forName("GBK"));
    23. int c;
    24. while ((c = fr.read())!=-1){
    25. System.out.print((char)c);
    26. }
    27. fr.close();
    28. }
    29. }
  • 代码演示-转换流

    1. /**
    2. * 转换输入流,字符转换为字节
    3. * 转换输出流,字节转换为字符
    4. */
    5. public class MainClass2 {
    6. public static void main(String[] args) throws IOException {
    7. //采用转换流指定编码格式
    8. /* InputStreamReader isr = new InputStreamReader(new FileInputStream("day_12\\file\\demo10.txt"),"GBK");
    9. OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("day_12\\file\\demo10-2.txt"),"UTF-8");
    10. int ch;
    11. while ((ch = isr.read())!=-1){
    12. osw.write(ch);
    13. }
    14. isr.close();
    15. osw.close();*/
    16. //JDK11后FileWriter,支持指定编码格式
    17. FileReader fr = new FileReader("day_12\\file\\demo10.txt", Charset.forName("GBK"));
    18. FileWriter fw = new FileWriter("day_12\\file\\demo10-2.txt",Charset.forName("UTF-8"));
    19. int c;
    20. while ((c = fr.read())!=-1){
    21. fw.write(c);
    22. }
    23. fr.close();
    24. fw.close();
    25. }
    26. }

6.对象操作流

可以把对象以字节的形式写到本地文件,直接打开文件,是读不懂的,需要再次用到对象操作流到内存中

6.1 对象序列化流【应用】

  • 对象序列化介绍
    • 对象序列化:就是将对象保存到磁盘中,或者在网络中传输对象
    • 这种机制就是使用一个字节序列表示一个对象,该字节序列包含:对象的类型、对象的数据和对象中存储的属性等信息
    • 字节序列写到文件之后,相当于文件中持久保存了一个对象的信息
    • 反之,该字节序列还可以从文件中读取回来,重构对象,对它进行反序列化
  • 对象序列化流: ObjectOutputStream
    • 将Java对象的原始数据类型和图形写入OutputStream。 可以使用ObjectInputStream读取(重构)对象。 可以通过使用流的文件来实现对象的持久存储。 如果流是网络套接字流,则可以在另一个主机上或另一个进程中重构对象
  • 构造方法 | 方法名 | 说明 | | —- | —- | | ObjectOutputStream(OutputStream out) | 创建一个写入指定的OutputStream的ObjectOutputStream |

  • 序列化对象的方法 | 方法名 | 说明 | | —- | —- | | void writeObject(Object obj) | 将指定的对象写入ObjectOutputStream |

  • 示例代码
    学生类

    1. public class Student implements Serializable {
    2. private String name;
    3. private int age;
    4. public Student() {
    5. }
    6. public Student(String name, int age) {
    7. this.name = name;
    8. this.age = age;
    9. }
    10. public String getName() {
    11. return name;
    12. }
    13. public void setName(String name) {
    14. this.name = name;
    15. }
    16. public int getAge() {
    17. return age;
    18. }
    19. public void setAge(int age) {
    20. this.age = age;
    21. }
    22. @Override
    23. public String toString() {
    24. return "Student{" +
    25. "name='" + name + '\'' +
    26. ", age=" + age +
    27. '}';
    28. }
    29. }
  • 测试类

    1. public class ObjectOutputStreamDemo {
    2. public static void main(String[] args) throws IOException {
    3. //创建一个写入指定的OutputStream的ObjectOutputStream
    4. ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("myOtherStream\\oos.txt"));
    5. //创建对象
    6. Student s = new Student("佟丽娅",30);
    7. //将指定的对象写入ObjectOutputStream
    8. oos.writeObject(s);
    9. //释放资源
    10. oos.close();
    11. }
    12. }
  • 注意事项

    • 一个对象要想被序列化,该对象所属的类必须必须实现Serializable 接口
    • Serializable是一个标记接口,实现该接口,不需要重写任何方法

6.2 对象反序列化流【应用】

  • 对象反序列化流: ObjectInputStream
    • ObjectInputStream反序列化先前使用ObjectOutputStream编写的原始数据和对象
  • 构造方法 | 方法名 | 说明 | | —- | —- | | ObjectInputStream(InputStream in) | 创建从指定的InputStream读取的ObjectInputStream |

  • 反序列化对象的方法 | 方法名 | 说明 | | —- | —- | | Object readObject() | 从ObjectInputStream读取一个对象 |

  • 示例代码

    1. public class ObjectInputStreamDemo {
    2. public static void main(String[] args) throws IOException, ClassNotFoundException {
    3. //ObjectInputStream(InputStream in):创建从指定的InputStream读取的ObjectInputStream
    4. ObjectInputStream ois = new ObjectInputStream(new FileInputStream("myOtherStream\\oos.txt"));
    5. //Object readObject():从ObjectInputStream读取一个对象
    6. Object obj = ois.readObject();
    7. Student s = (Student) obj;
    8. System.out.println(s.getName() + "," + s.getAge());
    9. ois.close();
    10. }
    11. }

6.3 serialVersionUID&transient【应用】

  • serialVersionUID
    • 用对象序列化流序列化了一个对象后,假如我们修改了对象所属的类文件,读取数据会不会出问题呢?
      • 会出问题,会抛出InvalidClassException异常
    • 如果出问题了,如何解决呢?
      • 重新序列化
      • 给对象所属的类加一个serialVersionUID
        • private static final long serialVersionUID = 42L;
  • transient
    • 如果一个对象中的某个成员变量的值不想被序列化,又该如何实现呢?
      • 给该成员变量加transient关键字修饰,该关键字标记的成员变量不参与序列化过程
  • 示例代码
    学生类

    1. public class Student implements Serializable {
    2. private static final long serialVersionUID = 42L;
    3. private String name;
    4. private transient int age;
    5. public Student() {
    6. }
    7. public Student(String name, int age) {
    8. this.name = name;
    9. this.age = age;
    10. }
    11. public String getName() {
    12. return name;
    13. }
    14. public void setName(String name) {
    15. this.name = name;
    16. }
    17. public int getAge() {
    18. return age;
    19. }
    20. public void setAge(int age) {
    21. this.age = age;
    22. }
    23. @Override
    24. public String toString() {
    25. return "Student{" +
    26. "name='" + name + '\'' +
    27. ", age=" + age +
    28. '}';
    29. }
    30. }
  • 测试类

    1. public class ObjectStreamDemo {
    2. public static void main(String[] args) throws IOException, ClassNotFoundException {
    3. // write();
    4. read();
    5. }
    6. //反序列化
    7. private static void read() throws IOException, ClassNotFoundException {
    8. ObjectInputStream ois = new ObjectInputStream(new FileInputStream("myOtherStream\\oos.txt"));
    9. Object obj = ois.readObject();
    10. Student s = (Student) obj;
    11. System.out.println(s.getName() + "," + s.getAge());
    12. ois.close();
    13. }
    14. //序列化
    15. private static void write() throws IOException {
    16. ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("myOtherStream\\oos.txt"));
    17. Student s = new Student("佟丽娅", 30);
    18. oos.writeObject(s);
    19. oos.close();
    20. }
    21. }

6.4 对象操作流练习【应用】

  • 案例需求
    创建多个学生类对象写到文件中,再次读取到内存中
  • 实现步骤
    • 创建序列化流对象
    • 创建多个学生对象
    • 将学生对象添加到集合中
    • 将集合对象序列化到文件中
    • 创建反序列化流对象
    • 将文件中的对象数据,读取到内存中
  • 代码实现
    学生类

    1. public class Student implements Serializable{
    2. private static final long serialVersionUID = 2L;
    3. private String name;
    4. private int age;
    5. public Student() {
    6. }
    7. public Student(String name, int age) {
    8. this.name = name;
    9. this.age = age;
    10. }
    11. public String getName() {
    12. return name;
    13. }
    14. public void setName(String name) {
    15. this.name = name;
    16. }
    17. public int getAge() {
    18. return age;
    19. }
    20. public void setAge(int age) {
    21. this.age = age;
    22. }
    23. }
  • 测试类

    1. /**
    2. * 序列化练习
    3. */
    4. public class MainClass {
    5. public static void main(String[] args) throws Exception {
    6. //write();
    7. read();
    8. }
    9. private static void read() throws IOException, ClassNotFoundException {
    10. ObjectInputStream ois = new ObjectInputStream(new FileInputStream("day_12\\file\\a.txt"));
    11. ArrayList<Student> list2 = (ArrayList<Student>) ois.readObject();
    12. for (Student student : list2) {
    13. System.out.println(student);
    14. }
    15. ois.close();
    16. }
    17. private static void write() throws Exception {
    18. /**
    19. * 如果要序列化的对象有多个,不建议直接将多个对象序列化到文件中,因为反序列化时容易出异常
    20. * 建议: 将要序列化的多个对象存储到集合中,然后将集合序列化到文件中
    21. */
    22. Student s1 = new Student("杜子腾",16);
    23. Student s2 = new Student("张三",23);
    24. Student s3 = new Student("李四",24);
    25. ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("day_12\\file\\a.txt"));
    26. ArrayList<Student> list = new ArrayList<>();
    27. list.add(s1);
    28. list.add(s2);
    29. list.add(s3);
    30. //我们往本地文件中写的就是一个集合
    31. oos.writeObject(list);
    32. oos.close();
    33. }
    34. }

7.Properties集合

7.1 Properties作为Map集合的使用【应用】

  • Properties介绍
    • 是一个Map体系的集合类
    • Properties可以保存到流中或从流中加载
    • 属性列表中的每个键及其对应的值都是一个字符串

7.2 Properties基本使用

  1. public class PropertiesDemo01 {
  2. public static void main(String[] args) {
  3. //创建集合对象
  4. Properties prop = new Properties();
  5. //存储元素
  6. prop.put("itheima001", "佟丽娅");
  7. prop.put("itheima002", "赵丽颖");
  8. prop.put("itheima003", "刘诗诗");
  9. //遍历集合
  10. Set<Object> keySet = prop.keySet();
  11. for (Object key : keySet) {
  12. Object value = prop.get(key);
  13. System.out.println(key + "," + value);
  14. }
  15. }
  16. }

7.3 Properties作为Map集合的特有方法【应用】

  • 特有方法 | 方法名 | 说明 | | —- | —- | | Object setProperty(String key, String value) | 设置集合的键和值,都是String类型,底层调用 Hashtable方法 put | | String getProperty(String key) | 使用此属性列表中指定的键搜索属性 | | Set stringPropertyNames() | 从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串 |

  • 示例代码 ```java /**

    • 特有方法 */ public class MainClass { public static void main(String[] args) {

      1. //Object setProperty(String key, String value) --- put
      2. //设置集合的键和值,都是String类型,底层调用 Hashtable方法 put
      3. Properties prop = new Properties();
      4. prop.setProperty("江苏","南京");
      5. prop.setProperty("安徽","南京");
      6. prop.setProperty("山东","济南");
      7. System.out.println(prop);
      8. System.out.println("---------------------");
      9. //String getProperty(String key) --- get
      10. //使用此属性列表中指定的键搜索属性

      String value = prop.getProperty(“江苏”);

      1. System.out.println(value);
      2. System.out.println("--------------------");
  1. //Set<String> stringPropertyNames() --- keySet
  2. //从该属性列表中返回一个不可修改的键集,其中键及其对应的值是字符串
  3. Set<String> keys = prop.stringPropertyNames();
  4. for (String key : keys) {
  5. String val = prop.getProperty(key);
  6. System.out.println(key + "=" + val);
  7. }
  8. }

}

  1. <a name="6ca29cf0"></a>
  2. ### 7.4 Properties和IO流相结合的方法【应用】
  3. - 和IO流结合的方法
  4. | 方法名 | 说明 |
  5. | --- | --- |
  6. | void load(Reader reader) | 从输入字符流读取属性列表(键和元素对) |
  7. | void store(Writer writer, String comments) | 将此属性列表(键和元素对)写入此 Properties表中,以适合使用 load(Reader)方法的格式写入输出字符流 |
  8. - 示例代码-load方法的使用
  9. ```java
  10. /**
  11. * Properties 加载文件
  12. */
  13. public class MainClass {
  14. public static void main(String[] args) throws IOException {
  15. Properties prop = new Properties();
  16. FileReader fr = new FileReader("day_12\\file\\prop.properties");
  17. //将本地文件中的键值对数据读取到集合中
  18. //调用完了load方法之后,文件中的键值对数据已经在集合中了.
  19. prop.load(fr);
  20. fr.close();
  21. System.out.println(prop);
  22. }
  23. }
  • 示例代码-store方法的使用
    1. /**
    2. * Properties 写入文件
    3. */
    4. public class MainClass2 {
    5. public static void main(String[] args) throws IOException {
    6. Properties prop = new Properties();
    7. prop.setProperty("username","zhangsan");
    8. prop.setProperty("password","123");
    9. prop.setProperty("db","mysql");
    10. FileWriter fw = new FileWriter("day_12\\file\\prop-db.properties");
    11. //将集合中的数据以键值对形式保存在本地
    12. prop.store(fw,"db config");
    13. fw.close();
    14. System.out.println(prop);
    15. }
    16. }

7.5 Properties集合练习【应用】

  • 案例需求
    在Properties文件中手动写上姓名和年龄,读取到集合中,将该数据封装成学生对象,写到本地文件
  • 实现步骤
    • 创建Properties集合,将本地文件中的数据加载到集合中
    • 获取集合中的键值对数据,封装到学生对象中
    • 创建序列化流对象,将学生对象序列化到本地文件中
  • 代码实现
    学生类

    1. public class Student implements Serializable {
    2. private static final long serialVersionUID = 1L;
    3. private String name;
    4. private int age;
    5. public Student() {
    6. }
    7. public Student(String name, int age) {
    8. this.name = name;
    9. this.age = age;
    10. }
    11. public String getName() {
    12. return name;
    13. }
    14. public void setName(String name) {
    15. this.name = name;
    16. }
    17. public int getAge() {
    18. return age;
    19. }
    20. public void setAge(int age) {
    21. this.age = age;
    22. }
    23. @Override
    24. public String toString() {
    25. return "Student{" +
    26. "name='" + name + '\'' +
    27. ", age=" + age +
    28. '}';
    29. }
    30. }
  • 测试类

    1. public class Test {
    2. public static void main(String[] args) throws IOException {
    3. //1.创建Properties集合,将本地文件中的数据加载到集合中
    4. Properties prop = new Properties();
    5. FileReader fr = new FileReader("prop.properties");
    6. prop.load(fr);
    7. fr.close();
    8. //2.获取集合中的键值对数据,封装到学生对象中
    9. String name = prop.getProperty("name");
    10. int age = Integer.parseInt(prop.getProperty("age"));
    11. Student s = new Student(name,age);
    12. //3.创建序列化流对象,将学生对象序列化到本地文件中
    13. ObjectOutputStream oos = new ObjectOutputStream(new FileOutputStream("a.txt"));
    14. oos.writeObject(s);
    15. oos.close();
    16. }
    17. }