• 深入理解Java IO流(第一篇)


    ✅作者简介:大家好我是@每天都要敲代码,一位材料转码农的选手,希望一起努力,一起进步!
    📃个人主页:@每天都要敲代码的个人主页
    💬推荐一款模拟面试、刷题神器,从基础到大厂面试题👉点击跳转刷题网站进行注册学习

    目录

    🥅IO流理论概述

    1.什么是IO

    2.IO流的分类

    3.流的四大家族

    4.需要掌握的十六个流

    🥅字节输入流FileInputStream

    1.FileInputStream初步理解

    2.FileInputStream常用方法

    🥅字节输出流FileOutputStream

    🥅任意文件拷贝

    🥅FileReader && FileWriter && 普通文件拷贝

    1.字符输入流FileReader

    2.字符输出流FileWriter

    3.普通文件拷贝

    结束语


    🥅IO流理论概述

    1.什么是IO

    ❤️什么是IO?IO有什么用?

          ⭐️I代表Input,把硬盘里的文件放到内存里,就叫做输入(Input),也就是

          ⭐️O代表Output,把内存里的文件放到硬盘里,就叫做输出(Output),也就是

    ❤️通过IO流可以完成文件的读和写!并且读和写都是以内存为参照的!

    2.IO流的分类

    ❤️IO流的分类?有多种分类方式:
        ⭐️第一种方式是按照流的方向进行分类(以内存作为参照物):            
        (1)往内存中去,叫做输入(Input)。或者叫做读(Read)。
        (2)从内存中出来,叫做输出(Output)。或者叫做写(Write)。
        ⭐️另一种方式是按照读取数据方式不同进行分类:
        (1)有的流是按照字节的方式读取数据,一次读取1个字节byte,等同于一次读取8个二进制位。这种流是万能的,什么类型的文件都可以读取。包括:文本文件,图片,声音文件,视频文件等....
         例:假设文件file1.txt,采用字节流的话是这样读的:
                 a中国bc张三fe
                 第一次读:一个字节,正好读到'a'
                 第二次读:一个字节,正好读到'中'字符的一半。
                 第三次读:一个字节,正好读到'中'字符的另外一半。
       (2)有的流是按照字符的方式读取数据,一次读取一个字符,这种流是为了方便读取
       普通文本文件而存在的,这种流不能读取:图片、声音、视频等文件。只能读取纯文本文件,连word文件都无法读取。
       例:假设文件file1.txt,采用字符流的话是这样读的:
               a中国bc张三fe
               第一次读:'a'字符('a'字符在windows系统中占用1个字节。)
               第二次读:'中'字符('中'字符在windows系统中占用2个字节。)
      ⭐️综上所述:流的分类
            (1)输入流、输出流
            (2)字节流、字符流

    3.流的四大家族

    ❤️Java中的IO流都已经写好了,我们最主要还是掌握,在java中已经提供了哪些流,每个流的特点是什么,每个流对象上的常用方法有哪些
        ⭐️java中所有的流都是在:java.io.*;

        ⭐️ java中主要还是研究:怎么new流对象?调用流对象的哪个方法是读?哪个方法是写?

    ❤️java IO流这块有四大家族
        ⭐️四大家族的四个首领:(都是抽象类(abstract class))
            java.io.InputStream          ​​​​字节输入流(读)
            java.io.OutputStream       字节输出流(写)

            java.io.Reader                ​​​​  字符输入流(读)
            java.io.Writer                    字符输出流(写)

       ⭐️所有的都实现了:

               (1)java.io.Closeable接口,都是可关闭的,都有close()方法
               (2) 流,毕竟是一个管道,这个是内存和硬盘之间的通道,用完之后一定要关闭,
                 不然会耗费(占用)很多资源。养成好习惯,用完流一定要关闭。

       ⭐️所有的输出流都实现了:
               (1) java.io.Flushable接口,都是可刷新的,都有flush()方法。
               (2)养成一个好习惯,输出流在最终输出之后,一定要记得flush()刷新一下,
                 这个刷新表示将通道/管道当中剩余未输出的数据强行输出完(清空管道),
               (3)注意:  刷新的作用就是清空管道;如果没有flush()可能会导致丢失数据。
    ❤️注意:在java中只要“类名”以Stream结尾的都是字节流。以“Reader/Writer”结尾的都是字符流。

    4.需要掌握的十六个流

    ❤️java.io包下需要掌握的流有16个:

     ⭐️文件专属:
            java.io.FileInputStream(掌握)
            java.io.FileOutputStream(掌握)
            java.io.FileReader
            java.io.FileWriter

     ⭐️转换流:(将字节流转换成字符流)
            java.io.InputStreamReader
            java.io.OutputStreamWriter

     ⭐️缓冲流专属:
            java.io.BufferedReader
            java.io.BufferedWriter
            java.io.BufferedInputStream
            java.io.BufferedOutputStream

     ⭐️数据流专属:
            java.io.DataInputStream
            java.io.DataOutputStream

     ⭐️标准输出流:
            java.io.PrintWriter
            java.io.PrintStream(掌握)

     ⭐️对象专属流:
            java.io.ObjectInputStream(掌握)
            java.io.ObjectOutputStream(掌握)

    🥅字节输入流FileInputStream

    1.FileInputStream初步理解

    ⭐️java.io.FileInputStream:
        1、文件字节输入流,万能的,任何类型的文件都可以采用这个流来读。
        2、字节的方式,完成输入的操作,完成读的操作(硬盘---> 内存)

        3、调用read()方法进行读,返回的是int类型(字符对应的ASCII码);没有元素的话,返回的是-1

    1. package com.bjpowernode.java.io;
    2. import java.io.FileInputStream;
    3. import java.io.FileNotFoundException;
    4. import java.io.IOException;
    5. public class FileInputStreamTest01 {
    6. public static void main(String[] args) {
    7. // 创建文件字节输入流对象
    8. // 文件路径://C:\Java学习\temp.txt
    9. //(里面存的是abc,IDEA会自动把\编程\\,因为java中\表示转义)
    10. // FileInputStream这个方法会抛出异常,父类是Exception,属于编译时异常,需要处理
    11. FileInputStream fis = null; //写到外面,主要是为了finally里面能够调用
    12. try { //C:/Java学习/temp.txt,路径写成这样也是可以的
    13. fis = new FileInputStream("C:\\Java学习\\temp.txt");
    14. //1、读文件,从此输入流中读取一个字节
    15. int readDate = fis.read();
    16. System.out.println(readDate); //97
    17. readDate = fis.read();
    18. System.out.println(readDate); //98
    19. readDate = fis.read();
    20. System.out.println(readDate); //99
    21. readDate = fis.read();
    22. System.out.println(readDate); //-1;最终没数据了,就返回-1
    23. //2、循环读
    24. while(true){
    25. int readDate1 = fis.read();
    26. if(readDate1 == -1){ // 没有数据返回的是-1
    27. break;
    28. }
    29. System.out.println(readDate1);
    30. }
    31. //3、优化while
    32. int readDate1 = 0;
    33. while((readDate1 = fis.read()) != -1){
    34. System.out.println(readDate1);
    35. }
    36. } catch (FileNotFoundException e) {
    37. e.printStackTrace();
    38. } catch (IOException e) { //read时候,补充的异常处理
    39. e.printStackTrace();
    40. } finally {
    41. // 加上finally关键字,无论最终有没有异常,都需要关闭这个流
    42. // 在finally语句块当中确保流一定关闭
    43. if (fis != null) { //生成这个的快捷键ifn
    44. // 关闭流的前提,流不是null;是空没必要关闭
    45. try {
    46. fis.close(); //有异常,try...catch
    47. } catch (IOException e) {
    48. e.printStackTrace();
    49. }
    50. }
    51. }
    52. }
    53. }

    ⭐️分析上面这个程序的缺点
    (1)一次读取一个字节byte,这样内存和硬盘交互太频繁,基本上时间/资源都耗费在交互上面了。所以能不能一次读取多个字节呢?答案是可以的。

    (2)int read(byte[] b) 一次最多读取 b.length 个字节
        减少硬盘和内存的交互,提高程序的执行效率;往byte[]数组当中读。

    (3)这里我们不在使用绝对路径,而是相对路径;那么IDEA默认的路径是什么呢?

            工程Project的根就是IDEA的默认当前路径;

    1. package com.bjpowernode.java.io;
    2. import java.io.FileInputStream;
    3. import java.io.FileNotFoundException;
    4. import java.io.IOException;
    5. public class FileInputStreamTest02 {
    6. public static void main(String[] args) {
    7. FileInputStream fis = null;
    8. try {
    9. // 我们写成相对路径方式,在这里要先理解在IDEA默认的当前路径是哪里?
    10. // 工程Project的根就是IDEA的默认当前路径;例如我的:C:\Users\86177\IdeaProjects\JavaSe1
    11. //fis = new FileInputStream("tempfile.txt"); //这是在工程下面的tempfile.txt
    12. // 如果是day06模块下的src包下有一个tempfile.txt,怎么调用呢?
    13. fis = new FileInputStream("day06/src/tempfile.txt"); //存放的是abcdef
    14. // 开始读,采用byte数组,一次读取多个字节。最多读取“数组.length”个字节。
    15. byte[] bytes = new byte[4]; 准备一个4个长度的byte数组,一次最多读取4个字节
    16. /* 1、普通打印
    17. int readCount = fis.read(bytes); // 返回的是当前的读取到的字节数量。(不是字节本身)
    18. System.out.println(readCount); // 4;第一次读到了4个字节
    19. //System.out.println(new String(bytes)); //abcd,将字符数组全部转换成字符串
    20. // 实际上应该读到多少个,就转换多少个
    21. System.out.println(new String(bytes,0,readCount)); // abcd
    22. readCount = fis.read(bytes);
    23. System.out.println(readCount); // 2;第二次只能读取到2个字节
    24. //System.out.println(new String(bytes)); //efcd,将字符数组全部转换成字符串,这里就出了问题
    25. System.out.println(new String(bytes,0,readCount)); //ef
    26. readCount = fis.read(bytes);
    27. System.out.println(readCount); //1个字节都没有读取到返回-1*/
    28. //2、 写成循环
    29. while(true){
    30. int readCount = fis.read(bytes);
    31. if(readCount == -1){
    32. break;
    33. }
    34. System.out.print(new String(bytes,0,readCount)); // abcdef
    35. }
    36. //3、 代码优化
    37. int readCount = 0;
    38. while((readCount = fis.read(bytes)) != -1){
    39. // 将读取到的将字符数组全部转换成字符串
    40. System.out.print(new String(bytes,0,readCount)); // abcdef
    41. }
    42. } catch (FileNotFoundException e) {
    43. e.printStackTrace();
    44. } catch (IOException e) {
    45. e.printStackTrace();
    46. }finally{
    47. if (fis != null) {
    48. try {
    49. fis.close();
    50. } catch (IOException e) {
    51. e.printStackTrace();
    52. }
    53. }
    54. }
    55. }
    56. }

    2.FileInputStream常用方法

    ❤️FileInputStream类的其它常用方法:

          ⭐int available()返回流当中剩余的没有读到的字节数量

          ⭐long skip(long n)跳过几个字节不读

    1. package com.bjpowernode.java.io;
    2. import java.io.FileInputStream;
    3. import java.io.FileNotFoundException;
    4. import java.io.IOException;
    5. public class FileInputStreamTest03 {
    6. public static void main(String[] args) {
    7. FileInputStream fis = null;
    8. try {
    9. fis = new FileInputStream("tempfile.txt"); // 存的abcdef
    10. // 先读一个字节
    11. int readByte = fis.read();
    12. System.out.println(readByte); // 97
    13. //1、available方法,还剩下多少字节
    14. System.out.println(fis.available()); // 5,还剩下5个字节
    15. // available方法有什么用?
    16. // 我们知道当前有的字节数,就不用使用循环了,直接指定当前长度就行
    17. // 这种方式不太适合太大的文件,因为byte[]数组不能太大。
    18. byte[] bytes = new byte[fis.available()]; //不需要循环,直接读一次就行
    19. int readCount = fis.read(bytes);
    20. System.out.println(new String(bytes,0,readCount));
    21. // 2、skip方法,跳过几个字节
    22. fis = new FileInputStream("tempfile.txt");
    23. // 跳过3个字节
    24. fis.skip(3);
    25. System.out.println(fis.read()); //100(d)
    26. } catch (FileNotFoundException e) {
    27. e.printStackTrace();
    28. } catch (IOException e) {
    29. e.printStackTrace();
    30. } finally {
    31. try {
    32. fis.close();
    33. } catch (IOException e) {
    34. e.printStackTrace();
    35. }
    36. }
    37. }
    38. }

    🥅字节输出流FileOutputStream

    ❤️文件字节输出流,从内存到硬盘,负责写;怎么写呢?

     ⭐如果当前文件不存在,会自动创建!

     ⭐如果当前文件已经存在,会把原来文件的内容进行清空覆盖!

     ⭐后面的参数直接跟的是byte数组

     ⭐如果是字符串,字符串转字节数组需要用getBytes()方法,把字符串转换成byte数组;而把byte数组转换成字符串用的是new String(byte数组)       

    1. package com.bjpowernode.java.io;
    2. import java.io.FileNotFoundException;
    3. import java.io.FileOutputStream;
    4. import java.io.IOException;
    5. public class FileOutputStreamTest01 {
    6. public static void main(String[] args) {
    7. FileOutputStream fos = null;
    8. try {
    9. //1、 myfile文件如果不存在的时候会自动新建!
    10. // 这种方式谨慎使用,会先将原文件清空,然后重新写入。
    11. fos = new FileOutputStream("myfile");
    12. //2、以追加的方式在文件末尾写入。不会清空原文件内容。
    13. fos = new FileOutputStream("myfile",true);
    14. //3、开始写,写到数组里
    15. byte[] bytes = {97,98,99,100};
    16. fos.write(bytes); // 写进去abcd
    17. //4、将byte数组的一部分写出
    18. fos.write(bytes,0,2); //写进去ab
    19. //5、写一个字符串,然后把字符串转换成byte数组
    20. String s = "我是一个中国人!";
    21. //将一个字符串转换成byte数组
    22. byte[] byts = s.getBytes();
    23. fos.write(byts); //写进去“我是一个中国人!”
    24. //6、写完之后一定要刷新
    25. fos.flush();
    26. } catch (FileNotFoundException e) {
    27. e.printStackTrace();
    28. } catch (IOException e) {
    29. e.printStackTrace();
    30. } finally {
    31. if (fos != null) {
    32. try {
    33. fos.close();
    34. } catch (IOException e) {
    35. e.printStackTrace();
    36. }
    37. }
    38. }
    39. }
    40. }

    🥅任意文件拷贝

    有了上面的输入流和输出流学习,我们是不是就可以尝试完成一个文件拷贝的功能,无非就是读、写的结合应用,下面我们先看一下原理图:

    ⭐使用FileInputStream + FileOutputStream完成文件的拷贝

    ⭐拷贝的过程应该是一边读,一边写

    ⭐使用字节流拷贝文件的时候,文件类型随意,什么样的文件都能拷贝。

    1. package com.bjpowernode.java.io;
    2. import java.io.FileInputStream;
    3. import java.io.FileNotFoundException;
    4. import java.io.FileOutputStream;
    5. import java.io.IOException;
    6. public class CopyTest {
    7. public static void main(String[] args) {
    8. FileInputStream fis = null;
    9. FileOutputStream fos = null;
    10. try {
    11. //1、 创建一个输入流对象
    12. fis = new FileInputStream("C:\\Java学习\\javaSE学习\\2.JavaSE初学习笔记\\2.第一章:Java环境搭建\\HelloWorld.java");
    13. //2、 创建一个输出流对象
    14. fos = new FileOutputStream("C:\\Java学习\\HelloWorld.java");
    15. //3、 最核心的部分:一边读,一边写
    16. byte[] bytes = new byte[1024 * 1024]; // 1MB(一次最多拷贝1MB。)
    17. int readCount = 0;
    18. while((readCount = fis.read(bytes)) != -1) {
    19. fos.write(bytes, 0, readCount);
    20. }
    21. //4、 刷新,输出流最后要刷新
    22. fos.flush();
    23. } catch (FileNotFoundException e) {
    24. e.printStackTrace();
    25. } catch (IOException e) {
    26. e.printStackTrace();
    27. } finally {
    28. // 分开try,不要一起try。
    29. // 一起try的时候,其中一个出现异常,可能会影响到另一个流的关闭。
    30. if (fos != null) {
    31. try {
    32. fos.close();
    33. } catch (IOException e) {
    34. e.printStackTrace();
    35. }
    36. }
    37. if (fis != null) {
    38. try {
    39. fis.close();
    40. } catch (IOException e) {
    41. e.printStackTrace();
    42. }
    43. }
    44. }
    45. }
    46. }

    🥅FileReader && FileWriter && 普通文件拷贝

    1.字符输入流FileReader

    (1)对于字符输入输出流FileReader 和 FileWriter的用法,与字节输入输出流FileInputStream FileOutputStream的用法很相似;后者我们已经学过使用byte数组,前者是使用char数组

    (2)FileReader: 文件字符输入流,只能读取普通文本。 读取文本内容时,比较方便,快捷。

    1. package com.bjpowernode.java.io;
    2. import java.io.FileNotFoundException;
    3. import java.io.FileReader;
    4. import java.io.IOException;
    5. public class FileReaderTest01 {
    6. public static void main(String[] args) {
    7. FileReader reader = null;
    8. try {
    9. // 创建文件字符输入流
    10. reader = new FileReader("C:\\Java学习\\javaSE学习\\2.JavaSE初学习笔记\\2.第一章:Java环境搭建\\HelloWorld.java");
    11. // 开始读
    12. char[] chars = new char[4]; // 一次读取4个字符
    13. // 第一种方法
    14. int readCount = 0;
    15. while((readCount = reader.read(chars)) != -1){
    16. System.out.print(new String(chars,0,readCount));
    17. }
    18. // 补充
    19. // reader.read(chars); // 往char数组中读
    20. // 按照字符的方式读取,一次读取一个字符
    21. for(char c :chars){
    22. System.out.println(c);
    23. }
    24. } catch (FileNotFoundException e) {
    25. e.printStackTrace();
    26. } catch (IOException e) {
    27. e.printStackTrace();
    28. } finally {
    29. if (reader != null) {
    30. try {
    31. reader.close();
    32. } catch (IOException e) {
    33. e.printStackTrace();
    34. }
    35. }
    36. }
    37. }
    38. }

    2.字符输出流FileWriter

    (1)FileWriter:文件字符输出流,写;只能输出普通文本!例如:图片、声音、视频、word文件等,都不可以!

    (2)后面的参数直接跟的是char数组;也可以直接跟字符串

    1. package com.bjpowernode.java.io;
    2. import java.io.FileWriter;
    3. import java.io.IOException;
    4. public class FileWriterTest01 {
    5. public static void main(String[] args) {
    6. FileWriter writer = null;
    7. try {
    8. //1、 创建文件字符输出流对象
    9. writer = new FileWriter("file",true); //file,没有会自动创建
    10. //2、 开始写
    11. char[] chars = {'我','是','中','国','人'};
    12. // 写整个数组的内容
    13. writer.write(chars);
    14. // 也可以只写数组的一部分
    15. writer.write(chars,0,2);
    16. // 后面也可以直接跟字符串
    17. writer.write("\n"); // 换行
    18. writer.write("我很骄傲");
    19. //3、 刷新
    20. writer.flush();
    21. } catch (IOException e) {
    22. e.printStackTrace();
    23. }finally{
    24. if (writer != null) {
    25. try {
    26. writer.close();
    27. } catch (IOException e) {
    28. e.printStackTrace();
    29. }
    30. }
    31. }
    32. }
    33. }

    3.普通文件拷贝

    使用FileReader 和 FileWriter进行拷贝,只能拷贝“普通文本”文件(能用记事本编辑的)!

    1. package com.bjpowernode.java.io;
    2. import java.io.*;
    3. public class CopyTest02 {
    4. public static void main(String[] args) {
    5. FileReader reader = null;
    6. FileWriter writer = null;
    7. {
    8. try {
    9. // 创建字符输入流
    10. reader = new FileReader("file");
    11. // 创建字符输出流
    12. writer = new FileWriter("myfile");
    13. // 边读边写
    14. char[] chars = new char[1024*1024]; // 1MB
    15. int readCount = 0;
    16. while((readCount = reader.read(chars)) != -1){
    17. writer.write(chars,0,readCount);
    18. }
    19. // 刷新
    20. writer.flush();
    21. } catch (FileNotFoundException e) {
    22. e.printStackTrace();
    23. } catch (IOException e) {
    24. e.printStackTrace();
    25. }finally {
    26. if (reader != null) {
    27. try {
    28. reader.close();
    29. } catch (IOException e) {
    30. e.printStackTrace();
    31. }
    32. }
    33. if(writer != null){
    34. try {
    35. writer.close();
    36. } catch (IOException e) {
    37. e.printStackTrace();
    38. }
    39. }
    40. }
    41. }
    42. }
    43. }

    结束语

    今天的分享就到这里啦!快快通过下方链接注册加入刷题大军吧!各种大厂面试真题在等你哦!

     💬刷题神器,从基础到大厂面试题👉点击跳转刷题网站

  • 相关阅读:
    MeterSphere 之 Idea插件开发
    Node.js 零基础入门 Node.js 零基础入门第四天 4.4 在项目中操作MySQL
    哪些人需要考PMP®证书呢?考了PMP®证书有什么用呢?
    【漏洞复现】74cms任意文件读取
    Java基础教学 日期类Api导学
    【c++刷题Day2】专题3栈与队列&单调栈与单调队列T4
    【uniapp】uview1.x 的 u-upload 上传点击删除隐藏 modal 提示框
    HJZS-E002(断电延时)电源监视继电器
    【大模型系列】问答理解定位(Qwen-VL/Llama2/GPT)
    跨境电商系统源码分享,助力企业快速搭建电商平台
  • 原文地址:https://blog.csdn.net/m0_61933976/article/details/125850482