字节流(InputStream和OutputStream),字节流读写文件,字节流的缓冲区,字节缓冲流

字节流
  • 抽象类InputStream和OutputStream是字节流的顶级父类
  • 所有的字节输入流都继承自InputStream,所有的输出流都继承子OutputStream

InputStream的常用方法:

int read()                                   
int read(byte b[])
int read(byte b[], int off, int len)    
void close()        

OutputStream的常用方法:

void write(int b)
void write(byte b[])
void write(byte b[], int off, int len)    
void flush() 
void close()    

注意:InputStream和OutputStream是抽象类,不能被实例化

InputStream和OutputStream提供不同的子类,这些子类的体系结构图如下:

在这里插入图片描述

字节流读写文件

概述:

在操作文件时,最常见的就是从一个文件中读取数据并将数据写入到另一个文件,这一过程就是文件的读写,针对文件的读写操作,JDK专门提供两个类,分别是FileInputStream和FileOutputStream

FileInputStream是InputStream的子类,操作文件的字节输入流,专门用于读取文件中的数据

创建一个text.txt文件,在文件中输入"hello"并保存,然后创建一个读取文件的类,代码如下:

package com.Put;

import java.io.FileInputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建一个文件字节输入流来读取文件
        FileInputStream in = new FileInputStream("text.txt");
        //定义一个int类型的变量
        int b = 0;
        //通过循环来读取文件,当返回值为-1结束循环
        while ((b=in.read())!=-1){
            System.out.print(b+" ");
            // System.out.print((char) b+" "); //h e l l o 
        }
        //关闭流
        in.close();
    }
    //运行结果:104 101 108 108 111 
}

简述:

在"text.txt"文件中,字符’h’ ‘e’ ‘l’ ‘l’ 'o’各占一个字节,因此最终结果显示的是文件"text.txt"中的5个字节所对应的ASCLL码值

需要注意的是必须保证文件在相应的目录下存在并且是可读,否则或抛出java.io.FileNotFoundException异常,即文件找不到异常

FileOutputStream是OutputStream的子类,操作文件的字节输出流,用于把数据写入文件

FileOutputStream写入数据时,自动创建了文件out.txt,并自定义的字符串写入到目标文件中,代码如下:

package com.Put;

import java.io.FileOutputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建文件输出流,并指定文件名称
        FileOutputStream out = new FileOutputStream("out.txt");
        //定义一个字符串
        String str = "hello";
        //将字符串转换为字节数组进行写入操作
        out.write(str.getBytes());
        //关闭流
        out.close();
    }
}

此外,如果通过FileOutputStream向一个已经存在的文件写入数据,那么该文件的数据首先会被清空,再写入新的数据;如果希望再已存在的文件内容后追加新内容,则可以使用FileOutStream的构造函数FileOutputStream(String fileName,boolean append)来创建文件输出流对象,并把append参数的值设置为true,代码如下:

package com.Put;

import java.io.FileOutputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建文件输出流,并指定文件名称
        FileOutputStream out = new FileOutputStream("out.txt",true);
        //定义一个字符串
        String str = "world";
        //将字符串转换为字节数组进行写入操作
        out.write(str.getBytes());
        //关闭流
        out.close();
    }
}

注意:如果append参数为false,存在的数据依然会被清空

文件的拷贝
  • 文件的拷贝需要通过输入流来读取源文件的数据,并通过输出流将数据写入新的文件(I/O流成对出现,即输入输出流一起使用)
package com.Put;

import java.io.FileInputStream;
import java.io.FileOutputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建文件输入流对象读取指定目录下的文件
        FileInputStream in = new FileInputStream("E:\\Idea\\JavaSE\\重新学习\\resource\\985.png");
        //创建文件输出流对象将读取到的文件内容写到指定目录的文件下
        FileOutputStream out = new FileOutputStream("E:\\Idea\\JavaSE\\重新学习\\target\\211.png");
        //定义一个int类型的变量
        int len = 0;
        //获取拷贝文件前的系统时间
        long beginTime = System.currentTimeMillis();
        //循环将读取到的文件字节信息写入到新文件
        while ((len=in.read())!=-1){
            out.write(len);
        }
        //获取拷贝后之后的系统时间
        long endTime = System.currentTimeMillis();
        //输出拷贝花费时间
        System.out.println((endTime-beginTime)+"毫秒");
        in.close();
        out.close();
    }
}

简述:

拷贝过程通过while循环将字节逐个进行拷贝,每循环一次,通过FileInputStream的read()方法读取一个字节,并通过FileOutputStream的write()方法将该字节写入指定文件,循环往复,直到读取的长度len为-1,表示读取到文件的末尾,结束循环。文件我使用的是绝对路径(相对绝对都可)

效果展示:

在这里插入图片描述

在这里插入图片描述

在这里插入图片描述

程序结束耗费地时间:

在这里插入图片描述

字节流的缓冲区

通过流的方法拷贝文件时,为了提高传输效率,自定义一个数组作为缓冲区,可以一次性读取多个字节的数据,将数据保存在字节数组中,然后将字节数组的数据一次性写入到新文件中,代码如下:

package com.Put;

import java.io.FileInputStream;
import java.io.FileOutputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建文件输入流对象读取指定目录下的文件
        FileInputStream in = new FileInputStream("E:\\Idea\\JavaSE\\重新学习\\resource\\985.png");
        //创建文件输出流对象将读取到的文件内容写到指定目录的文件下
        FileOutputStream out = new FileOutputStream("E:\\Idea\\JavaSE\\重新学习\\target\\211.png");
        //定义一个int类型的变量
        int len = 0;
        //定义一个长度为1024的字节数组
        byte[] buff = new byte[1024];           //!!!!!!!!!!!!!!!!
        //获取拷贝文件前的系统时间
        long beginTime = System.currentTimeMillis();
        //循环将读取到的文件字节信息写入到新文件 
        while ((len=in.read(buff))!=-1){       //!!!!!!!!!!!!!!!!!!
            out.write(buff,0,len);
        }
        //获取拷贝后之后的系统时间
        long endTime = System.currentTimeMillis();
        //输出拷贝花费时间
        System.out.println((endTime-beginTime)+"毫秒");
        in.close();
        out.close();
    }
}

简述:在拷贝过程中,使用while循环语句逐渐实现字节文件的拷贝,每循环一次,就从文件读取若干字节填充到数组,并通过变量len记住读入数组的字节数,然后从数组的第一个字节开始,将len个字节写入到新文件。直到len的值为-1,说明已经读到了文件的末尾,循环结束,整个拷贝过程就结束了,最终程序会将整个文件拷贝到目标文件。

运行时间:

在这里插入图片描述

字节缓冲流
  • 两个带缓冲的字节流,分别是BufferedInputStream和BufferedOutputStream,他们的构造方法中分别接收InputStream和OutputStream类型的参数作为对象,在读取时提供缓冲区功能。

应用程序、缓冲流和底层字节流的关系,如图:

在这里插入图片描述

从图可以看出,应用程序时通过缓冲流来完成数据读写的,而缓冲流又是通过底层字节流与设备进行关联的。

BufferedInputStream和BufferedOutputStream这两个缓冲流的用法如下:

package com.Put;

import java.io.BufferedInputStream;
import java.io.BufferedOutputStream;
import java.io.FileInputStream;
import java.io.FileOutputStream;

public class IO {
    public static void main(String[] args) throws Exception{

        //创建输入输出流字节缓冲流对象
        BufferedInputStream bis = new BufferedInputStream(new FileInputStream("E:\\Idea\\JavaSE\\重新学习\\resource\\985.png"));
        BufferedOutputStream bos = new BufferedOutputStream(new FileOutputStream("E:\\Idea\\JavaSE\\重新学习\\target\\211.png"));
        //定义一个int类型的变量
        int len = 0;
        //获取拷贝文件前的系统时间
        long beginTime = System.currentTimeMillis();
        //通过循环读取输入字节缓冲流中的数据,并通过输出字节缓冲流写入到新文件
        while ((bis.read())!=-1) {
            bos.write(len);
        }
        //获取拷贝后之后的系统时间
        long endTime = System.currentTimeMillis();
        //输出拷贝花费时间
        System.out.println((endTime-beginTime)+"毫秒");
        //关闭流
        bis.close();
        bos.close();
        
    }
}

运行结果图:

在这里插入图片描述

简述:

创建了BufferedInputStream和BufferedOutputStream这两个缓冲流对象,这两个流内部都定义了一个大小为8192的字节数组,当定义read()和write()方法读取数据时,首先将读写的数据存入到定义的好的字节数组,然后将字节数组一次性读写到文件中。这种方式与字节流的缓冲区相似,都对数据进行了缓冲,提高数据读取的效率。

  • 6
    点赞
  • 17
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

等慢慢

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

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

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

打赏作者

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

抵扣说明:

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

余额充值