黑马Java—第18讲 字节缓冲流&字符流

124 阅读14分钟

❤️持续创作,加速成长!这是我参与「掘金日新计划 · 10 月更文挑战」的第3天,点击查看活动详情

🎨 个人介绍

👉大家好,我是:旺仔不是程序员

👉认真分享技术,记录学习过程的点滴,如果我的分享能为你带来帮助,请支持我奥🍻

👉你的支持,是我每天更新的动力。

👉赞点:👍 留言:✍ 收藏:⭐

👉个人格言:想法一步一步的落实,才是你我前进最佳选择。

1.字节缓冲流

1.1 字节缓冲流

1 )字节缓冲流介绍

  1. BufferedOutStream : 该类实现缓冲输出流。通过设置这样的输出流,应用程序可以向底层输出流写如字节,而不必为写入每一个字节导致底层系统的调用
  2. BufferedInputStream: 创建BufferedInputStream 将创建一个内部缓冲区数组。当从流中读取或跳过字节时,内部缓冲区将根据需要从所包含堵塞输入流中重新填充,一次很多字节

2 )构造方法

方法名说明
BUfferedoutputStream(OutputStream out)创建字节缓冲输出流对象
BufferedInputStream(InputStream in)创建字节缓冲输入流对象

3 )代码实例

public static void main(String[] args) throws IOException {
    // 字节缓冲输出流:BufferedOutStream
    BufferedOutputStream bout = new BufferedOutputStream(new FileOutputStream("E:\BackEndLearning\work\Test1\box.txt"));
​
    // 写数据
    bout.write("hello\r\n".getBytes(StandardCharsets.UTF_8));
    bout.write("world\r\n".getBytes(StandardCharsets.UTF_8));
​
    //释放资源
    bout.close();
​
    //字节输入流
    BufferedInputStream in = new BufferedInputStream(new FileInputStream("E:\BackEndLearning\work\Test1\box.txt"));
​
    // 一次读取一个字节的数组
    byte[] bys = new byte[1024];
    int len;
    while((len = in.read(bys))!=-1) {
        System.out.println(new String(bys,0,len));
    }
​
    //释放资源
    in.close();
}

1.2 字节流复制视频

1 )案例需求

  1. 把C:\Users\86131\Desktop\1-27\1-27\JAVA第11讲.mp4 复制到 E:\itCast\JAVA第11讲.mp4

2 )实现步骤

  1. 根据数据源创建字节输入流对象
  2. 根据目的地创建字节输出流对象
  3. 读写数据,复制视频
  4. 释放资源

3 )代码实现

public static void main(String[] args) {
    long startTime = currentTimeMillis();
    method1();
    long endTime = System.currentTimeMillis();
​
    System.out.println("共耗时" + (endTime-startTime) + "毫秒");
​
}
//用缓冲流实现视频的复制
private static void method1() {
    try {
        // 缓冲输入流
        BufferedInputStream bin = new BufferedInputStream(new FileInputStream("C:\Users\86131\Desktop\1-27\1-27\JAVA第11讲.mp4"));
        //缓冲输出流
        BufferedOutputStream out = new BufferedOutputStream(new FileOutputStream("E:\itCast\JAVA第11讲.mp4"));
​
        byte[] bytes = new byte[1024];
        int len;
        while ((len = bin.read(bytes)) != -1) {
            out.write(bytes, 0, len);
        }
​
​
    }catch (IOException e) {
        e.printStackTrace();
    }
​
}

2. 字符流

2.1 为什么会出现字符流

1 )字符流的介绍

  1. 由于字节流操作中文不是特别的方便,所以Java 就提供了字符流
  2. 字符流 = 字节流 + 编码表

2 )中文的字节存储方式

  1. 用字节流复制文本文件时,文本文件也会有中文,但是没有问题,原因是最终底层操作会自动进行字符拼接
  2. 汉字在存储的时候,无论选择那种编码存储,第一个字节都是负数

2.2 编码表

1 )什么时字符集

  1. 是一个系统的所有字符的集合,包括各国家文字,标点符号,图形符号,数字等
  2. 计算机要准确的存储和识别各种字符集符号,就需要进行字符编码,一套字符集必然至少有一套字符编码
  3. 常见字符集有ASCII字符集,GBXXX字符集,Unicode字符集

2 )常见的字符集合

  1. ASCII字符集:

    • lASCII:是基于拉丁字母的一套电脑编码系统,用于显示现代英语,主要包括控制字符(回车键、退格、换行键等)和可显示字符(英文大小写字符、阿拉伯数字和西文符号)
    • 基本的ASCII字符集,使用7位表示一个字符,共128字符。ASCII的扩展字符集使用8位表示一个字符,共256字符,方便支持欧洲常用字符。
    • 是一个系统支持的所有字符的集合,包括各国家文字、标点符号、图形符号、数字等
  2. GBXXX字符集:

    • GBK:最常用的中文码表
    • 是在GB2312标准基础上的扩展规范,使用了双字节编码方案,共收录了21003个汉字,完全兼容GB2312标准,同时支持繁体汉字以及日韩汉字等
  3. Unicode字符集:

    • UTF-8编码:可以用来表示Unicode标准中任意字符,它是电子邮件、网页及其他存储或传送文字的应用中,优先采用的编码。互联网工程工作小组(IETF)要求所有互联网协议都必须支持UTF-8编码

    • 一至四个字节为每个字符编码

    • 编码规则:

      • 128个US-ASCII字符,只需一个字节编码
      • 拉丁文等字符,需要二个字节编码
      • 大部分常用字(含中文),使用三个字节编码
      • 其他极少使用的Unicode辅助字符,使用四字节编码

2.3 字符串中的编码解码问题

1 )相关方法

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

2 )代码演示

 public static void main(String[] args) throws UnsupportedEncodingException {
     // 定义一个字符串
     String s = "中国";
     byte[] bytes = s.getBytes(StandardCharsets.UTF_8);  //[-28, -72, -83, -27, -101, -67]
     byte[] gbks = s.getBytes("GBK");  //[-42, -48, -71, -6]
     System.out.println(Arrays.toString(bytes));
     System.out.println(Arrays.toString(gbks));
​
     System.out.println(new String(bytes));
     System.out.println(new String(gbks,"GBK"));
 }

2.4 字符流中的编码解码问题

1 )字符流中和编码解码问题相关的两个类

  1. InputStreamReader:是从字节流到字符流的桥梁

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

    • 是从字符流到字节流的桥梁,使用指定的编码将写入的字符编码为字节
    • 它使用的字符集可以由名称指定,也可以被明确指定,或者可以接受平台默认字符集

2 )构造方法

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

3 )代码演示

public static void main(String[] args) throws IOException {
    //创建字符流对象
    OutputStreamWriter osw = new OutputStreamWriter(new FileOutputStream("E:\BackEndLearning\work\Test1\osw.txt"));
​
    // 写数据
    osw.write("中国");
    osw.close();
​
    //读取数据
    InputStreamReader reader = new InputStreamReader(new FileInputStream("E:\BackEndLearning\work\Test1\osw.txt"));
​
    //一次读取一个字符数据
    int ch;
    while((ch=reader.read())!=-1) {
        System.out.println((char)ch);
    }
    reader.close();
}

2.5 字符流写数据的5种方式

1 )方法介绍

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

2 )刷新和关闭的方法

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

3 )代码实例

public static void main(String[] args) throws IOException {
    OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream("E:\BackEndLearning\work\Test1\osw.txt"));
​
    //写数据
    writer.write("中国");
    writer.close();
​
    //读取数据
    InputStreamReader reader = new InputStreamReader(new FileInputStream("E:\BackEndLearning\work\Test1\osw.txt"));
​
    //一次读取一个字符数据
    int ch;
    while((ch=reader.read())!=-1) {
        System.out.print((char)ch);
    }Out
​
    //释放资源
    reader.close();
}
  1. 第二部分代码
public static void main(String[] args) throws IOException {
     //字符写数据流
     OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream("E:\BackEndLearning\work\Test1\oaw.txt"));
​
     // 写一个字符
     writer.write(97);
     writer.write(98);
     writer.write(99);
     writer.write(100);
     writer.write("\r\n");
​
​
     //写一个数组
     char[] chs = {'a', 'b', 'c', 'd', 'e'};
     writer.write(chs);
     writer.write("\r\n");
​
     //写入字符数组的一部分
     writer.write(chs,0,chs.length);
     writer.write("\r\n");
     writer.write(chs,0,3);
     writer.write("\r\n");
​
     //写入一个字符串
     writer.write("abcdefgh");
     writer.write("\r\n");
     //写一个字符串的一部分
     writer.write("abcdefgh",0,3);
     writer.write("abcdefgh",0,"abcdefgh".length());
​
     writer.flush();
​
     //关闭资源
     writer.close();
 }

2.6 字符读取数据的两种方式

1 )方法介绍

方法名说明
int read()一次读取一个字符数据
int read(char([] cbuf)一次读一个字符数组

2 )代码演示

 public static void main(String[] args) throws IOException {
     InputStreamReader reader = new InputStreamReader(new FileInputStream("E:\BackEndLearning\work\Test1\oaw.txt"));
​
     //一次读取一个字符的数据
     int ch; //因为读取返回的是byte类型,所以是int
     while((ch=reader.read()) != -1) {
         System.out.print((char) ch);
     }
​
     //一次读一个字符数组数据
     char[] chs = new char[1024];
     int len;
     while((len=reader.read(chs)) != -1) {
         System.out.print(new String(chs,0,len));
     }
​
     //释放资源
     reader.close();
 }

2.7 字符流复制java文件

1 )案例需求

  1. 把E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java 复制到 E:\itCast\Student.java

2 )步骤实现

  1. 根据数据源创建字符输入流对象
  2. 根据目的地创建字符输出流对象
  3. 读写数据,复制文件
  4. 释放资源

3 )代码实现

public static void main(String[] args) throws IOException {
    //根据数据源创建输入流对象
    InputStreamReader reader = new InputStreamReader(new FileInputStream("E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java"));
​
    // 根据目的地创建输出流对象
    OutputStreamWriter writer = new OutputStreamWriter(new FileOutputStream("E:\itCast\Student.java"));
​
    //一次读写一个字符数据
    int ch;
    while((ch=reader.read())!=-1) {
        writer.write(ch);
    }
​
    //一次读写一个数组数据
    char[] chs = new char[1024];
    int len;
    while((len=reader.read(chs))!=-1) {
        writer.write(chs,0,len);
    }
​
    //释放资源
    reader.close();
    writer.close();
}

2.8 字符缓冲流

1 )字符缓冲流介绍

  1. BufferedWriter :将文本写入字符输出流,缓冲字符,以提供单个字符,数组和字符串的高效写入,可以指定缓冲区大小,或者可以接受默认大小。默认值足够大,可用于大多数用途
  2. BufferedReader :从字符输入流读取文本,缓冲字符,以提供字符,数组和行的高效读取,可以指定缓冲区的大小,或者可以使用默认大小。默认值足够大,可用于大多数用途。

2 )构造方法

方法名说明
BufferedWriter(Writer out)创建字符缓冲输出流对象
BufferedReader(Reader in)创建字符缓冲输入流对象

3 )代码演示

public static void main(String[] args) throws IOException{
    //创建字符缓冲输出流
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\BackEndLearning\work\Test1\bw.txt"));

    writer.write("hello\r\n");
    writer.write("world\r\n");

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

    //创建字符缓冲字符输入流
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\bw.txt"));

    // 一次读取一个字符数据
    int ch;
    while ((ch = reader.read()) != -1) {
        System.out.print((char) ch);
    }

    //一次读取一个数组的数据
    char[] chs = new char[1024];
    int len;
    while ((len = reader.read(chs)) != -1) {
        System.out.print(new String(chs,0,len));
    }
    //释放资源
    reader.close();
}

注意:

  1. 读取数据不会重复读取,即,上边两种方式去读数据,只能显示一次的读取数据,不能重复显示

2.9 字符缓冲路复制Java文件

1 )案例需求

  1. 把E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java 复制到 E:\itCast\Student.java

2 )实现步骤

  1. 根据数据源创建字符缓冲输入流对象
  2. 根据目的地创建字符缓冲输出流对象
  3. 读写数据,复制文件,使用字符缓冲流特有功能实现
  4. 释放资源

3 )代码实现

public static void main(String[] args) throws IOException {
    //根据数据源创建字符缓冲输入流对象
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java "));

    //根据目的地创建字符缓冲输出流对象
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\itCast\Student.java"));

    //一次读写一个字符的数据
    int ch;
    while ((ch = reader.read()) != -1) {
        writer.write(ch);
    }

    // 一次读写一个字符数组数据
    char[] chs = new char[1024];
    int len;
    while ((len = reader.read(chs)) != -1) {
        writer.write(chs, 0, len);
    }

    //释放资源
    reader.close();
    writer.close();
}

2.10 字符缓冲流特有的功能

1 )方法介绍

  1. BufferedWriter :
方法名说明
void newLine()写一行 行分隔符,行分隔符字符串由系统属性定义
  1. BufferedReader
方法名说明
String readLine()读一行文字,结果包含行的内容的字符串,不包括任何行终止字符如果流的结尾已经到达,则为null

3 )代码演示

public static void main(String[] args) throws IOException{
    //创建字符缓冲输出流
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\BackEndLearning\work\Test1\bw.txt"));

    // 写数据
    for (int i = 0; i <10; i++) {
        writer.write("hello" + i);
        writer.newLine();
        writer.flush();
    }

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

    //创建字符缓冲输入流
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\bw.txt"));

    String line;
    while((line=reader.readLine())!=null) {
        System.out.println(line);
    }

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

2.11 字符缓冲流特有的功能复制Java文件

1 )案例需求

  1. 使用特有的功能,把把E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java 复制到 E:\itCast\Student.java

2 )实现步骤

  1. 根据数据源创建字符缓冲输入流对象
  2. 根据目的地创建字符缓冲输出流对象
  3. 读写数据,复制文件,使用字符缓冲流特有功能实现
  4. 释放资源

3 )代码实现

public static void main(String[] args) throws IOException {
    //根据数据源创建字符缓冲输入流对象
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\src\cn\heima\pro\Student.java "));

    //根据目的地创建字符缓冲输出流对象
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\itCast\Student.java"));

    //读写数据,复制文件
    String line;
    while((line=reader.readLine())!=null) {
        writer.write(line);
        writer.newLine();
        writer.flush();
    }

    //释放资源
    reader.close();
    writer.close();
}

2.12 IO流 小结

1 )字节流

2 )字符流

3.练习案例

3.1 文件到集合

1 )案例需求

  1. 把文本文件中的数据读取到集合中,并遍历集合。要求:文件种每行数据是一个集合元素

2 )实现步骤

  1. 创建字符缓冲输入流对象
  2. 创建ArrayList集合对象
  3. 调用字符缓冲输入流对象的方法读取数据
  4. 把读取到的字符串数据存储到集合中
  5. 释放资源
  6. 遍历集合

3 )代码实现

public static void main(String[] args) throws IOException {
    //创建字符缓冲输入流对象
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\bw.txt"));

    //创建ArrayList集合对象
    ArrayList<String> array = new ArrayList<String>();

    //调用字符缓冲输入流对象的方法读取数据
    String line;
    while((line = reader.readLine())!=null) {
        // 把读取到的字符串数据存储到集合中
        array.add(line);
    }

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

    //遍历集合
    for(String s :array) {
        System.out.println(s);
    }
}

3.2 集合到文件

1 )案例需求

  1. 把ArrayList集合中的字符串数据写入到文本文件。要求:每个字符串元素作为文件中的一行数据

2 )实现步骤

  1. 创建ArrayList集合
  2. 往集合中存储字符串元素
  3. 创建字符缓冲输出流对象
  4. 遍历集合,得到每一个字符串数据
  5. 调用字符缓冲流输出对象的方法写数据
  6. 释放资源

3 )代码实现

public static void main(String[] args) throws IOException{
    //创建集合对象
    ArrayList<String> list = new ArrayList<String>();

    //往集合中存储字符串元素
    list.add("Hello");
    list.add("world");
    list.add("java");

    //创建字符缓冲输出瑞对象
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\BackEndLearning\work\Test1\bw.txt",true));

    //遍历集合,得到每一个元素
    for(String s : list) {
        // 调用缓冲输出流对象的方法写数据
        writer.append(s);
        writer.newLine();
        writer.flush();
    }
    //释放资源
    writer.close();
}

3.3 点名器

1 )案例需求

  1. 我有一个文件里面存储了班级同学的姓名,每一个姓名占一行,要求通过程序实现随机点名器

2 )实现步骤

  1. 创建字符缓冲流对象
  2. 创建ArrayList集合对象
  3. 调用字符缓冲输入流对象的方法读数据
  4. 把读取到的字符串数据存储到集合中
  5. 释放资源
  6. 使用Random产生一个随机数,随机数的范围在:[0,集合的长度]
  7. 把第六步产生的随机数作为索引到ArrayList集合中获取值
  8. 把第七步得到的数据输出在控制台上

3 )代码实现

public static void main(String[] args) throws IOException {
    // 创建随机数类
    Random random = new Random();
    //创建缓冲字节输入流对象
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\bw.txt"));

    //创建ArrayList集合对象
    ArrayList<String> array = new ArrayList<>();

    //调用字符缓冲输入流对象方法读数据
    String line;
    while((line = reader.readLine())!=null) {
        array.add(line);
    }

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

    //产生随机数
    int index = random.nextInt(array.size());

    //把第六步产生的随机数作为索引 到ArrayList集合中获取值
    String s = array.get(index);

    // 获取获取到的值是
    System.out.println(s);
    System.out.println(index);
}

3.4 集合到文件改进版

1 )案例需求

  1. 把ArrayList集合中的学生数据写入到文本文件。要求:每一个学生对象的数据作为文件中的一行数 格式 :学号,姓名,年龄,居住地 举例:itheima001,林青霞,30,西安

2 )步骤实现

  1. 定义学生类
  2. 创建ArrayList集合
  3. 创建学生对象
  4. 把学生对象添加到 集合中
  5. 遍历集合,得到每一个学生对象
  6. 把学生对象的数据拼接成指定格式的字符串
  7. 调用字符串缓冲输出流对象的方法写数据
  8. 释放资源

3 )代码实现

  1. 学生类
public class Student {
    // 成员变量
    private String name;
    private int age;
    private String sId;
    private String address;

    public Student() {
    }


    public Student(String name, int age, String sId, String address) {
        this.name = name;
        this.age = age;
        this.sId = sId;
        this.address = address;
    }

    public String getsId() {
        return sId;
    }

    public void setsId(String sId) {
        this.sId = sId;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }

    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;
    }  
}
  1. 测试类
public static void main(String[] args) throws IOException {
    // 创建ArrayList集合
    ArrayList<Student> studentList = new ArrayList<Student>();

    //创建学生对象
    Student s1 = new Student("itheima001", "林青霞", 30, "西安");
    Student s2 = new Student("itheima002", "张曼玉", 35, "武汉");
    Student s3 = new Student("itheima003", "王祖贤", 33, "郑州");

    //把学生对象添加到集合中
    studentList.add(s1);
    studentList.add(s2);
    studentList.add(s3);

    //创建字符缓冲输出流对象
    BufferedWriter writer = new BufferedWriter(new FileWriter("E:\BackEndLearning\work\Test1\studentList.txt"));

    //遍历集合  ,得到每一个学生对象
    for(Student s : studentList) {
        //把学生对象拼接成指定格式的字符串
        StringBuilder sb = new StringBuilder();

        sb.append(s.getsId()).append(",").append(s.getName()).append(",").append(s.getAge()).append(",").append(s.getAddress());

        writer.write(sb.toString());
        writer.newLine();
        writer.flush();
    }

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

3.5 文件到集合的改进版

1 )案例需求

  1. 把文本文件的数据读取到集合中,并遍历集合。要求:文件中每一行数据是一个学生对象的成员变量值 举例:itheima001,林青霞,30,西安

2 )实现步骤

  1. 定义学生类
  2. 创建字符缓冲输入流对象
  3. 创建ArrayList 集合对象
  4. 调用字符缓冲输入流对象的方法读数据
  5. 把读取到的字符串数据用split()进行分割,得到一个字符串数组
  6. 创建学生对象
  7. 把字符串数组中的每一个元素取出来对应的赋值给学生对象的成员变量值
  8. 把学生对象添加到集合
  9. 释放资源
  10. 遍历集合

3 )代码实现

  1. 学生类,同上
  2. 测试类
public static void main(String[] args) throws IOException {
    //创建字符缓冲输入流
    BufferedReader reader = new BufferedReader(new FileReader("E:\BackEndLearning\work\Test1\studentList.txt"));

    //创建ArrayList集合
    ArrayList<Student> studentList = new ArrayList<Student>();

    // 调用字符缓冲输入流对象的方法读取数据
    String line;
    while ((line = reader.readLine()) != null) {
        //进行字符串的分割
        String[] strArray = line.split(",");

        //创建学生对象
        Student s = new Student();
        s.setsId(strArray[0]);
        s.setName(strArray[1]);
        s.setAge(Integer.parseInt(strArray[2]));
        s.setAddress(strArray[3]);

        //把学生对象添加到集合
        studentList.add(s);
    }

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

    //遍历集合
    for (Student s : studentList) {
        System.out.println(s.getsId() + "," + s.getName() + "," + s.getAge() + "," + s.getAddress());
    }
}

🎈看完了不妨给我点个赞吧,👉你的支持,是我每天更新的动力...