Java学习笔记(十)

Java学习笔记(十)

IO流

概述:Java中的I/O操作主要是指使用java.io包下的内容,进行输入、输出操作,输入也叫做读取数据,输出也叫做写入数据

IO的分类

根据数据的流向分为:输入流输出流

  • 输入流:把数据从其他设备上读取到内存中的流
  • 输出流:把数据从内存中写出到其他设备上的流

根据数据的类型分为:字符流字节流

  • 字节流:以字节为单位,读写数据的流
  • 字符流:以字符为单位,读取数据的流
    在这里插入图片描述

顶级父类

输入流输出流
字节流字节输入流
InputStream
字节输出流
OutputStream
字符流字符输入流
Reader
字符输出流
Writer

字节流

一切皆为字节

一切文件数据(文本、图片、视频等)在存储时,都是以二进制数字的形式保存,都是一个一个的字节,那么传输时一样如此。所以,字节流可以传输任意文件数据。在操作流的时候,我们需要时刻明确,无论使用什么样的流对象,底层传输的始终为二进制数据

字节输出流OutputStream

java.io.OutputStream抽象类是表示字节输出流的所有类的超类,将指定的字节信息写出到目的地。它定义了字节输出流的基本共性功能方法

  • public void close():关闭此输出流并释放与此流相关联的任何系统资源
  • public void flush():刷新此输出流并强制任何缓冲的输出字节被写入
  • public void write(byte[] b):将b.length字节从指定的字节数组写入此输出流
  • public void write(byte[] b,int off,int len):从指定的字节数组写入len字节,从偏移量off开始输出到此输出流
  • public abstract void write(int b):将指定的字节写入此输出流
FileOutputStream类

java.io.FileOutputStream类是OutputStream类的子类,是文件字节输出流,用于将数据写入文件

构造方法
  • public FileOutStream(File file):创建文件输出流写入由指定的File对象表示的文件
  • public FileOutputStream(String name):创建一个向具有指定名称的文件中写入数据的输出文件流
  • 参数:
    • String name:目的地是一个文件路径
    • File file:目的地是一个文件
  • 作用
    • 创建一个FileOutputStream对象
    • 会根据构造方法中传递的文件/文件路径,创建一个空的文件
    • 会把FileOutputStream对象指向创建好的文件
写入数据的原理(内存—>硬盘):
  • Java程序—>JVM(Java虚拟机)—>OS(操作系统)—>OS调用写数据的方法—>把数据写入文件中
/*使用步骤:
    1、创建一个FileOutputStream对象,构造方法中传递写入数据目的地
    2、调用FileOutputStream对象中的方法write把数据写入到文件中
    3、释放资源(流的使用会占用一些内存,使用完毕后把内存清空,提高程序效率)
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        // 1、创建一个FileOutputStream对象,构造方法中传递写入数据目的地
        FileOutputStream fos = new FileOutputStream("C:\\test\\a.txt");
        // 2、调用FileOutputStream对象中的方法write把数据写入到文件中
        fos.write(97);
        //3、释放资源(流的使用会占用一些内存,使用完毕后把内存清空,提高程序效率)
        fos.close();
    }
}

在这里插入图片描述

public class Demo {
    public static void main(String[] args) throws IOException {
        /*
            public void write(byte[] b):将`b.length`字节从指定的字节数组写入此输出流
            一次写多个字节
                如果第一个字节为正数(0-127)那么显示的时候会查询ASCII表
                如果第一个字节为负数,那么第一个字节和第二个字节,两个字节组成一个中文显示,查询系统默认的码表(GBK)
            public void write(byte[] b,int off,int len):把字节数组中的一部分写入文件中
                int off:数组开始的索引
                int len:写几个字节
            写入字符串的方法:可以使用String类中的getBytes方法把字符串转换为字节数组
         */
        FileOutputStream fos = new FileOutputStream(new File("C:\\test\\a.txt"));
        byte[] bytes = {65,66,67,68,69};//ABCDE
        //byte[] bytes = {-65,-66,-67,68,69};//烤紻E
        fos.write(bytes);
        fos.write(bytes,1,2);//BC
        byte[] bytes1 = "你好".getBytes();
        fos.write(bytes1);
        fos.close();
    }
}
数据的追加续写和换行

续写:

  • public FileOutputStream(File file,boolean append):创建文件按输出流以写入由指定的File对象表示的文件
  • public FileOutputStream(String name,boolean append):创建文件输出流以指定的名称写入文件
  • 这两个构造方法,参数中都需要传入一个boolean类型的值,true表示追加数据不会清空原有数据(覆盖原文件),false表示清空原有数据(覆盖原文件)

换行:写换行符号

  • windows:\r\n
  • linux:/n
  • mac:/r
字节输入流InputStream

java.io.InputStream抽象类是表示字节输入流的所有类的超类,可以读取字节信息到内存中,它定义了字节输入流的基本共性功能方法

  • public void close():关闭此输入流并释放与此流相关的任何系统资源
  • public abstract int read():从输入流读取数据到下一个字节
  • public int read(byte[] b):从输入流中读取一些字节数,并把他们存储到字节数组b中
FileInputStream类

java.io.FileInputStream类是InputStream类的子类,是文件字节输入流,从文件中读取字节

构造方法
  • public FileInputStream(File file):通过打开与实际文件的连接创建一个 FileInputStream ,该文件由文件系统中的 File对象 file命名。
  • public FileInputStream(String name):通过打开与实际文件的连接来创建一个 FileInputStream ,该文件由文件系统中的路径名 name命名。
  • 参数:读取文件的数据源
    • String name:文件的路径
    • File file:文件
  • 作用:
    • 会创建一个FileInputStream对象
    • 会把FileInputStream对象指定构造方法中要读取的文件
读取数据的原理(硬盘—>内存):
  • Java程序—>JVM—>OS—>OS读取数据的方法—>读取文件
/*使用步骤:
    1、创建FileInputStream对象,构造方法中绑定要读取的数据源
    2、使用FileInputStream对象中的方法read,读取文件
    3、释放资源
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        // 1、创建FileInputStream对象,构造方法中绑定要读取的数据源
        //a.txt中的内容为:abc
        FileInputStream fis = new FileInputStream("C\\test\\a.txt");
        // 2、使用FileInputStream对象中的方法read,读取文件
        // int len = fis.read();
        // System.out.println(len);//97 a
        // len = fis.read();
        // System.out.println(len);//98 b
        // len = fis.read();
        // System.out.println(len);//99 c
        // len = fis.read();
        // System.out.println(len);//-1 读取到文件末尾返回-1
        /*
            可以使用循环优化
            不知道文件中由多少字节,使用while循环
            while循环结束条件,读取到-1时候结束
        */
        int len = 0;
        while ((len = fis.read()) != -1){
            System.out.println(len);
        }
        //3、释放资源
        fis.close();
    }
}

在这里插入图片描述

/*
    字节输入流一次读取多个字节的方法
        int read(byte[] b)
        方法的参数byte[]的作用
            起到缓冲作用,存储每次读取的每个字节
            数组的长度一般定义为1024(1kb)或者1024的整数倍
        方法的返回值int是什么
            每次读取的有效字节个数
       String类的构造方法
        String(byte[] bytes):把字节数组转换成字符串
        String(byte[] bytes,int offset,int length):把字节数组的一部分转换为字符串
            offset:数组的开始索引
            length:转换的字节个数
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        //a.txt中的内容为ABCDE
        FileInputStream fis = new FileInputStream("C\\test\\a.txt");
        // byte[] bytes = new byte[2];
        // int len = fis.read(bytes);
        // System.out.println(len);//2
        // //System.out.println(Arrays.toString(bytes));//[65,66]
        // System.out.println(new String(bytes));//AB
        // len = fis. read(bytes);
        // System.out.println(len);//2
        // System.out.println(new String(bytes));//CD
        // len = fis. read(bytes);
        // System.out.println(len );//1
        // System.out.println(new String(bytes));//ED
        // len = fis. read(bytes);
        // System.out.println(len);//-1
        // System.out.println(new String(bytes));//ED
        /*
            可以使用循环优化
            不知道文件中由多少字节,使用while循环
            while循环结束条件,读取到-1时候结束
        */
        byte[] bytes = new byte[1024];
        int len = 0;
        while ((len = fis.read(bytes)) != -1){
            System.out.println(new String(bytes,0,len));
        }
        fis.close();
    }
}

在这里插入图片描述

字符流

当使用字节流读取文件的时候,可能会由一个问题,就是遇到中文字符时,可能不会显示完整的字符,那是因为一个中文字符可能占用多个字节存储,所以Java提供一些字符流类,以字符为单位读取数据,专门用于处理文本文件

/*
    使用字节流读取中文文件
    1个中文
        GBK:占用两个字节
        UTF-8:占用3个字节
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        //a.txt中的内容为你好
        FileInputStream fis = new FileInputStream("C\\test\\a.txt");
        int len = 0;
        while ((len = fis.read()) != -1){
            System.out.println(len);//228 189 160  229 165 189
            System.out.println((char) len);//乱码
        }
        fis.close();
    }
}

字符输入流Reader

java.io.Reader抽象类时表示用于读取字符流的所有类的超类,可以读取信息到内存中。它定义了字符输入流的基本共性功能方法

  • public void close():关闭刺溜并释放与次流相关联的任何系统资源
  • public int read():从输入流读取一个字符
  • public int read(char[] cbuf):从输入流中读取多个字符,并把它们存储到字符数组cbuf中
FileReader类

java.io.FileReader类是InputStreamReader类的子类(java.io.FileReader extends InputStreamReader extends Reader),是读取字符文件的类,构造时使用系统默认的字符编码和默认字节缓冲区,把硬盘中的数据以字符的方式读取到内存中

构造方法
  • public FileReader(String filename):创建一个新的 FileReader ,给定要读取的文件的名称。
  • public FileReader(File file):创建一个新的 FileReader ,给出 File读取。
  • 参数:读取文件的数据源
    • String filename:文件的路径
    • File file:一个文件
  • 作用:
    • 创建一个FileReader对象
    • 会把FileReader对象指向要读取的文件
/*
    字符输入流的使用步骤
        1、创建FileReader对象,构造方法中绑定要读取的数据源
        2、使用FileReader对象中的方法read读取文件
        3、释放资源
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        //1、创建FileReader对象,构造方法中绑定要读取的数据源
        //b.txt文件内容:你好abc123###
        FileReader fr = new FileReader("C:\\test\\b.txt");
        //2、使用FileReader对象中的方法read读取文件
        // int len = 0;
        // while ((len = fr.read()) != -1){
        //     System.out.println((char)len);//你好abc123###
        // }
        char[] chars = new char[1024];
        int len = 0;
        while ((len = fr.read()) != -1){
            /*
                String类的构造方法
                    String(char[] value):把字符数组转换为字符串
                    String(char[] value,int offset,int count):把字符数组的一部分转换为字符串
                        offset:数组开始的索引
                        count转换的个数
             */
            System.out.println(new String(chars, 0, len));//你好abc123###
        }
        //3、释放资源
        fr.close();
    }
}
字符输出流Writer

java.io.Writer抽象类时表示用于写出字符流的所有类的超类,将指定的字符信息写出到目的地,它定义了字节输出流的基本共性功能方法

  • void write(int c):写入单个字符
  • void write(char[] cbuf):写入字符数组
  • abstract void write(char[] cbuf,int off,int len):写入字符数组的某一部分,off是数组的开始索引,len是写的字符个数
  • void write(String str):写入字符串
  • void write(String str,int off,int len):写入字符串的某一部分,off是字符串的开始索引,len是写的字符个数
  • void flush():刷新该流的缓冲
  • void close():关闭此流,但要先刷新它
FileWriter类

java.io.FileWriter extends OutputStreamWriter extends WriterFileWriter类是文件字符输入流类,该类把内存中的字符数据写入到文件中

构造方法
  • public FileWriter(File file):根据给定的File对象构造一个FileWriter对象
  • public FileWriter(String filename):根据给定的文件名构造一个FileWriter对象
  • 参数:写入数据的目的地
    • String filename:文件的路径
    • File file:是一个文件
  • 作用:
    • 会创建一个FileWriter对象
    • 会根据构造方法中传递的文件/文件的路径,创建文件
    • 会把FileWriter对象指向创建好的文件
/*
    字符输出流的使用步骤
        1、创建FileWriter对象,构造方法中绑定要写入的数据目的地
        2、使用FileWriter中的方法write,把数据写入到内存缓冲区中(字符转换为字节的过程)
        3、使用FileWriter中的方法flush,把内存缓冲区中的数据,刷新到文件中
        4、释放资源(会先把内存缓冲区中数据刷新到文件中)
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        //1、创建FileWriter对象,构造方法中绑定要写入的数据目的地
        FileWriter fw = new FileWriter("C:\\test\\a.txt");
        //2、使用FileWriter中的方法write,把数据写入到内存缓冲区中(字符转换为字节的过程)
        fw.write(97);
        //3、使用FileWriter中的方法flush,把内存缓冲区中的数据,刷新到文件中
        //fw.flush();
        //4、释放资源(会先把内存缓冲区中数据刷新到文件中)
        fw.close();
    }
}
关闭和刷新

因为内置缓冲区的原因,如果不关闭输出流,无法写入字符到文件中,但是关闭的流对象,是无法继续写入数据的,如果我们既想写入数据,就需要flush方法

  • flush:刷新缓冲区,流对象可以继续使用
  • close:先刷新缓冲区,然后通知系统释放资源,流对象不可以再被使用
字符输出流写数据的其他方法
/*
    - `void write(char[] cbuf)`:写入字符数组
    - `abstract void write(char[] cbuf,int off,int len)`:写入字符数组的某一部分,off是数组的开始索引,len是写的字符个数
    - `void write(String str)`:写入字符串
    - `void write(String str,int off,int len)`:写入字符串的某一部分,off是字符串的开始索引,len是写的字符个数
 */
public class Demo {
    public static void main(String[] args) throws IOException {
        FileWriter fw = new FileWriter("C:\\test\\a.txt");
        //- `void write(char[] cbuf)`:写入字符数组
        char[] cs = {'a','b','c','d'};
        fw.write(cs);//abcd
        //- `abstract void write(char[] cbuf,int off,int len)`:写入字符数组的某一部分,off是数组的开始索引,len是写的字符个数
        fw.write(cs,1,3);//bcd
        //- `void write(String str)`:写入字符串
        fw.write("我是程序员");//我是程序员
        //- `void write(String str,int off,int len)`:写入字符串的某一部分,off是字符串的开始索引,len是写的字符个数
        fw.write("我是程序员",2,3);//程序员
        fw.close();
    }
}
数据的追加续写和换行

续写:

  • FileWriter(String filename,boolean append)
  • FileWriter(File file,boolen append)
  • 参数:
    • String filename,File file:写入数据的目的地
    • boolean append:续写开关,true不会创建新的文件覆盖原文件,可以续写;false创建新的文件覆盖原文件

换行:换行符号

  • windows:\r\n
  • linux:/n
  • mac:/r

IO异常的处理

  • JDK1.7之前使用try...catch...finally处理流中的异常

  • 格式:

    • try{
          //可能产生异常的代码
      }catch(异常类对象 变量名){
          异常的处理逻辑
      }finally{
          //一定会执行的代码
          //资源释放
      }
      
  • public class Demo {
        public static void main(String[] args) {
            FileWriter fw = null;
            try {
                fw = new FileWriter("C:\\test\\a.txt", true);
                for (int i = 0; i < 10; i++) {
                    fw.write("HelloWorld" + i + "\r\n");
                }
            } catch (IOException e) {
                e.printStackTrace();
            } finally {
               if (fw != null){
                   try {
                       fw.close();
                   } catch (IOException e) {
                       e.printStackTrace();
                   }
               }
            }
        }
    }
    
  • JDK1.7的新特性,在try的后边可以增加一个(),在括号中可以定义流对象,那么这个流对象的作用域就在try中有效,try中的代码执行完毕,会自动把流对象释放,不用再写finally

  • 格式:

    • try(定义流对象;定义流对象.....){
          //可能产生异常的代码
      }catch(异常类对象 变量名){
          //异常的处理逻辑
      }
      
  • public class Demo {
        public static void main(String[] args) {
            try(FileInputStream fis = new FileInputStream("C:\\1.jpg");
                FileOutputStream fos = new FileOutputStream("D:\\1.jpg");
                ){
                int len = 0;
                while ((len = fis.read()) != -1){
                    fos.write(len);
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }
    
  • JDK9的新特性,try的前边可以定义流对象,再try后边的()中可以直接引入流对象的名称(变量名),在try代码执行完毕之后,流对象也可以释放掉,不用写finally

  • 格式:

    • A a = new A();
      B b = new B();
      try(a;b){
          //可能产生异常的代码
      }catch(异常类对象 变量名){
          //异常的处理逻辑
      }
      
  • public class Demo {
        public static void main(String[] args) throws FileNotFoundException {
            FileInputStream fis = new FileInputStream("C:\\1.jpg");
            FileOutputStream fos = new FileOutputStream("D:\\1.jpg");
            try(fis;fos){
                int len = 0;
                while ((len = fis.read()) != -1){
                    fos.write(len);
                }
            }catch (IOException e){
                e.printStackTrace();
            }
        }
    }
    
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

掉发阿龙

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值