张小飞的Java之路——第四十二章——字节流

这篇具有很好参考价值的文章主要介绍了张小飞的Java之路——第四十二章——字节流。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

写在前面:

视频是什么东西,有看文档精彩吗?

视频是什么东西,有看文档速度快吗?

视频是什么东西,有看文档效率高吗?


诸小亮:下面我们学习——字节流

张小飞:什么是字节流?

诸小亮:就是以字节的形式操作文件中的数据,分为:InputStream、OutputStream

  • Stream:流,InputStream是输入流,OutputStream是输出流

张小飞:这个。。。,还是不太懂,能否演示一下?

FileOutputStream

诸小亮:好吧,我们先看——FileOutputStream

FileOutputStream:文件输出流,是 OutputStream 的子类,用于保存数据到文件

张小飞:您的意思是,可以利用 FileOutputStream 把数据存到文件中?

诸小亮:是的

张小飞:那么,具体怎么操作呢?

创建对象

诸小亮:首先,需要创建 FileOutputStream 对象,有两种方式?

张小飞:是给它一个 文件路径 吗?

诸小亮:不错,比如:


public static void main(String[] args) throws FileNotFoundException {

    //1. 创建输出流对象,如果指定路径上的文件不存在,会自动创建,如果存在会自动覆盖
    FileOutputStream out = new FileOutputStream("G:\\learn\\hero.txt");
}

张小飞:我这里怎么报错了呢?

张小飞的Java之路——第四十二章——字节流
诸小亮:这是因为你提供的路径上的文件夹不存在,所以抛出 FileNotFoundException

张小飞:哦~~,原来如此

诸小亮:除了根据 文件路径 创建对象之外,还可以根据 File 创建,比如:

FileOutputStream out = new FileOutputStream(new File("G:\\learn\\hero.txt"));

张小飞:这种方式,如果路径上的文件夹不存在,也会报错吗?

诸小亮:是的

存储数据

张小飞:那么,如何保存数据呢?

诸小亮:使用它的 write 方法写入字节数据,比如:

public static void main(String[] args) throws IOException {
    //1. 创建输出流对象
    FileOutputStream out = new FileOutputStream("G:\\learn\\hero.txt");

    //2. 调用写入功能
    String str = "yase,lvbu,liubei";
    //write:接收一个字节数组,把数组中的所有数据都写到输出流中
    out.write(str.getBytes());

    //3. 流是系统资源,需要释放,理论上应该放到finally中
    out.close();
}

诸小亮:运行程序后,你就会发现硬盘上会多一个 hero.txt 文件

张小飞:还真是的,好神奇

诸小亮:接下来,我们优化一下代码

public static void main(String[] args) {
    FileOutputStream out = null;
    try{
        //FileOutputStream 可能报异常,放到try中
        out = new FileOutputStream("G:\\learn\\hero.txt");
        String str = "yase,lvbu,liubei";
        out.write(str.getBytes());
    }catch (Exception e){
        e.printStackTrace();
    }finally {
        //创建 out 对象可能失败,所以要判断是不是 null;
        if(out!=null){
            try {
                out.close();
            } catch (IOException e) {
                //如果close失败,我们无法处理,直接抛出
                throw new RuntimeException("流关闭失败:"+e.getMessage());
                //注意:本人没见过close方法失败的,但还是得这样写
            }
        }
    }
}

张小飞:您这是优化代码吗?

诸小亮:是的,还记得我们说过嘛,流——是系统资源,使用过后要关闭

张小飞:记得,记得,不过您上面的代码也太麻烦了吧

诸小亮:这倒是,不过还可以再次优化

public static void main(String[] args) {
    //JDK7 开始,可以把 out 的定义放到 try 中,会自动 close
    try(FileOutputStream out = new FileOutputStream("G:\\learn\\hero.txt")){
        String str = "yase,lvbu,liubei";
        out.write(str.getBytes());
    }catch (Exception e){
        e.printStackTrace();
    }
}

张小飞:嗯,这样子就优雅多了

续写和换行

张小飞:上面的程序,每次执行程序后,发现文件中的内容都一样,不能往文件中追加内容吗?

诸小亮:目前上面代码,每次执行都会重新创建文件,所以内容都一样,当然可以追加内容,比如:

public static void main(String[] args) throws IOException {
    //1. 第二个参数是 append:true表示追加数据,false表示重新创建文件
    FileOutputStream out = new FileOutputStream("G:\\learn\\hero.txt", true);

    //2. 调用写入功能
    String str = "diaochan,change,xishi";
    //write:接收一个字节数组,把数组中的所有数据都写到输出流中
    out.write(str.getBytes());

    //3. liu是系统资源,需要释放,应该放到finally中
    out.close();
}

张小飞:嗯嗯,不错,再次执行后,内容追加成功了

张小飞的Java之路——第四十二章——字节流

张小飞:还有个问题,能不能写入数据的时候,换一行?

诸小亮:这个也没问题,加上 \r\n 就可以实现换行,比如:

张小飞的Java之路——第四十二章——字节流

结果:张小飞的Java之路——第四十二章——字节流

张小飞:明白了,我也来试试

诸小亮:不过,建议使用 System.lineSeparator()

张小飞:嗯?这是为什么呢?

诸小亮:\r\n 是 window 系统中的换行符,也就是说,当程序跑在 window 中才可以换行

如果是 linux 、Mac 就不行了,使用 System.lineSeparator(),可以让程序跨平台

张小飞的Java之路——第四十二章——字节流

结果:

张小飞的Java之路——第四十二章——字节流
张小飞:明白了

FileInputStream

诸小亮:接着是——FileInputStream

FileInputStream:文件输出流,是 InputStream 的子类,用于读取数据到内存中

张小飞:原来使用它读取文件数据

创建对象

诸小亮:FileInputStream创建对象的方式跟 FileOutputStream 一行,不再详细解释了

public static void main(String[] args) throws FileNotFoundException {
    //第一种:根据路径创建FileInputStream
    FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
    //第二种:根据File创建FileInputStream
//        FileInputStream in = new FileInputStream(new File("G:\\learn\\hero.txt"));
}

如果路径上的文件夹都不存在,则报错,比如:

张小飞的Java之路——第四十二章——字节流

结果:

张小飞的Java之路——第四十二章——字节流

读取数据

张小飞:它是如何读取数据的呢?

诸小亮:它读取的方式有好几种,我们一个个介绍

单字节读取

诸小亮:第一种,单字节读取

张小飞:这是什么意思?

诸小亮:它提供了一个 read 方法,但是每次只能读取一个字节,比如:

FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
//读取数据,每次只是读取一个字节,当 返回 -1 时,文件读取完毕
int read = in.read();
System.out.println(read);
//关闭流
in.close();

结果:张小飞的Java之路——第四十二章——字节流

张小飞:呀,结果怎么是一个数字?

诸小亮:多新鲜啊,字节是 byte ,这不是数字类型吗?

张小飞:噢~,差点儿忘记了,那为什么是 121 呢?

诸小亮:文件中第一个字母是 y,而 121 就是字母 y 对应的ASCII

张小飞:明白了,再把它转换为 char 类型,就行了吧

诸小亮:是的,把 int 转换为 char

张小飞的Java之路——第四十二章——字节流

结果:张小飞的Java之路——第四十二章——字节流

一次性读取

张小飞:这一次读取一个字节也太费劲了,有没有其他方式?

诸小亮:readAllBytes 可以一次读出所有内容,返回字节数组,比如:

FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
//readAllBytes:一次读出所有内容
byte[] bytes = in.readAllBytes();
//把字节数组转换问String
String content = new String(bytes);
System.out.println(content);
//关闭流
in.close();

结果:

张小飞的Java之路——第四十二章——字节流

张小飞:嗯嗯,这个好,以后就用这个吧

读取指定大小

诸小亮:一点儿都不好,如果文件太大,这样读取的话会占用很多内存,导致程序卡死的

张小飞:。。。。。,那怎么办?

诸小亮:它提供的 read 有重载的方法,可以一次读取指定大小

  • read(byte b[]):一次读取指定数量的字节,放到数组中,并返回读取的字节数量

张小飞:原来还有这种方法,怎么不早点说出来呢?

诸小亮:这不是让你也了解一下其他方法嘛

public static void main(String[] args) throws IOException {
    FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
    //指定一次读取 16 字节
    byte[] buffer = new byte[16];
    //read方法读取的内容都会放到buffer中,当返回 -1 时候,文件读取完毕
    int read = in.read(buffer);
    //把字节数组转换为String
    String content = new String(buffer);
    System.out.println(content);
    //关闭流
    in.close();
}

结果:张小飞的Java之路——第四十二章——字节流

张小飞:这才读取了 16 字节,得把文件中的内容都读出来啊

诸小亮:使用循环就行了

public static void main(String[] args) throws IOException {
    FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
    //指定一次读取16字节
    byte[] buffer = new byte[16];
    //read方法读取的内容都会放到buffer中,当返回-1时候,文件读取完毕
    int read = 0;
    while((read = in.read(buffer)) != -1){
        //把字节数组转换问String
        String content = new String(buffer);
        System.out.print(content);
    }
    //关闭流
    in.close();
}

结果:

张小飞的Java之路——第四十二章——字节流

张小飞:您这个输出,怎么多出来一些内容?

诸小亮:因为最后一次读出来的数据不够16字节,数组中还存留着上一次读出的内容,所以。。。。

张小飞:那,这样肯定不行啊,怎么解决?

诸小亮:放心,放心,read的返回值是每次读出来的字节数,可以用用它的返回值

张小飞的Java之路——第四十二章——字节流

诸小亮:上图,每次读出来几个字节就转换几个字节

结果:

张小飞的Java之路——第四十二章——字节流

张小飞:嗯嗯,这样好多了

缓冲区大小的设置

诸小亮:你知道我们使用缓冲区的目的是什么吗?

张小飞:不就是为了一次多读出一些数据吗?

诸小亮:其实准确来说,是减少读取文件的次数

张小飞:嗯?为什么?

诸小亮:因为读写文件是IO操作,需要操作硬盘,频繁的读写性能会很差

张小飞:那,目前缓冲区是 16 字节,如果一个文件太大,不是还会读很多次吗?

诸小亮:所以我们要合理设置缓冲区的大小

张小飞:怎么合理设置呢?

诸小亮:有个 available 方法 :返回流中剩余的未读的字节数量(其实就是文件大小)

张小飞:然后呢?

诸小亮:然后就可以根据我们的需求设置缓冲区了,比如:

张小飞的Java之路——第四十二章——字节流
按照上图中设置,最多读 6 次就可以读完

张小飞:那如果文件很大呢?比如 5 个 G

诸小亮:。。。。。,那这就要看你机器性能了,性能很高,就可以设置比较大的缓冲区

张小飞:哦,好吧

诸小亮:另外,还有一个获取文件大小的方法——length

张小飞的Java之路——第四十二章——字节流

复制文件

诸小亮:我们已经学习了 FileInputStream 、FileOutputStream,你能用它们做一个文件复制功能吗?

张小飞:这个。。。。,还不太行

诸小亮:好吧,先给你一个示例

public static void main(String[] args) throws IOException {
    //从源文件中读
    FileInputStream in = new FileInputStream("G:\\learn\\hero.txt");
    //目标文件
    FileOutputStream out = new FileOutputStream("G:\\hero.txt");
    //指定一次读取16字节
    byte[] buffer = new byte[16];
    int read = 0;
    while((read = in.read(buffer)) != -1){
        //每次只是写入这一次读取的字节数量
        out.write(buffer,0,read);
    }
    //关闭流
    in.close();
    out.close();
}

结果:

张小飞的Java之路——第四十二章——字节流

诸小亮:其中原理图都给你画好了

张小飞的Java之路——第四十二章——字节流

BufferedInputStream 和 BufferedOutputStream

诸小亮:接下来我们看看——BufferedInputStream、BufferedOutputStream

张小飞:这是做什么的?

诸小亮:它们是——字节流缓冲区对象,目的是增强 InputStream 和 OutputStream 的功能

张小飞:如何增强?

诸小亮:BufferedInputStreamBufferedOutputStream 内部有一个字节数组

张小飞:缓冲区?

诸小亮:是的,使用它们后,读写顺序是这样的

  • 先从源文件读数据到 **BufferedInputStream **的内部数组
  • 然后从 **BufferedInputStream **中再读到自定义的数组
  • 接着从自己定义的数组写到 BufferedOutputStream 的内部数组
  • 最后写到硬盘的新文件中

张小飞的Java之路——第四十二章——字节流

代码:

public static void main(String[] args) throws IOException {
    BufferedInputStream inBuffer = new BufferedInputStream(new FileInputStream("G:\\learn\\hero.txt"));
    BufferedOutputStream outBuffer = new BufferedOutputStream(new FileOutputStream("G:\\hero.txt"));
    byte[] buffer = new byte[16];
    int read = 0;
    //之前是从硬盘中读取,现在是从 BufferedInputStream 中读取
    while((read = inBuffer.read(buffer)) != -1){
        //之前直接写到硬盘上,现在写到 BufferedOutputStream 中
        outBuffer.write(buffer,0,read);
    }
    //关闭流
    inBuffer.close();
    inBuffer.close();
}

张小飞:是不是写错了,我运行了程序,发现文件中没有任何内容啊

诸小亮:这是因为 BufferedOutputStream 中的数组大小是 8192 字节,而我们文件 83 字节

  • 默认只有 BufferedOutputStream 中的数组满了后才会写到硬盘上

张小飞:那该怎么办?

诸小亮:可以手动调用 flush 方法,把 BufferedOutputStream 中的数据刷到硬盘上

张小飞的Java之路——第四十二章——字节流

张小飞:好的,我再试试

中文数据

张小飞:刚刚读写的都是英文,能不能弄点儿中文?

诸小亮:当然可以了

public static void main(String[] args) throws IOException {
    FileOutputStream out = new FileOutputStream("G:\\learn\\hero.txt");
    String str = "亚瑟";
    out.write(str.getBytes());
    out.close();
}

张小飞:我看了一下文件大小,默认用的是 UTF-8 编码吗?

张小飞的Java之路——第四十二章——字节流

诸小亮:用的什么编码,得看你自己在 idea 中怎么配置的

张小飞的Java之路——第四十二章——字节流

张小飞的Java之路——第四十二章——字节流文章来源地址https://www.toymoban.com/news/detail-464341.html

到了这里,关于张小飞的Java之路——第四十二章——字节流的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 小白到运维工程师自学之路 第四十六集 (mongodb复制集)

           1、 MongoDB复制集(MongoDB Replica Set)是MongoDB提供的一种高可用性和数据冗余的解决方案。它由多个MongoDB实例组成,其中一个作为主节点(Primary),其他节点则扮演从节点(Secondary)的角色。主节点处理所有的写操作和客户端请求,而从节点负责复制主节点的数据并

    2024年02月13日
    浏览(33)
  • 小白到运维工程师自学之路 第四十四集 (mariadb高可用集群故障转移)

            故障转移是指在集群中某个节点发生故障时,自动将服务转移到其他正常节点上的 过程。在MariaDB高可用集群中,通常使用主从复制的方式来实现故障转移。其中一个 节点被指定为主节点,负责处理所有的写操作和部分读操作,其他节点作为从节点,负 责复制主

    2024年02月11日
    浏览(30)
  • 小白到运维工程师自学之路 第四十九集 (正则表达式之grep)

    1、正则表达式(Regular Expression,简称为RegExp或Regex)是一种用于描述、匹配和操作文本的字符串模式的表达式。它提供了一种强大而灵活的方式来进行字符串的搜索、替换、提取和验证操作。 2、正则表达式可以用于各种编程语言和应用程序中,包括文本编辑器、命令行工具

    2024年02月13日
    浏览(36)
  • 学习java第四十三天

    Spring AOP 相关术语 (1)切面(Aspect):切面是通知和切点的结合。通知和切点共同定义了切面的全部内容。 (2)连接点(Join point):指方法,在Spring AOP中,一个连接点总是代表一个方法的执行。连接点是在应用执行过程中能够插入切面的一个点。这个点可以是调用方法时

    2024年04月15日
    浏览(30)
  • 学习java第四十一天

    Spring MVC 运行流程: 第一步:发起请求到前端控制器(DispatcherServlet) 第二步:前端控制器请求HandlerMapping查找 Handler( 可以根据xml配置、注解进行查找) 第三步:处理器映射器HandlerMapping向前端控制器返回Handler 第四步:前端控制器调用处理器适配器去执行Handler 第五步:处理

    2024年04月13日
    浏览(30)
  • 学习JAVA打卡第四十一天

    字符串与字符数组、字节数组 ⑴字符串与字符数组 String类的构造方法String(char a[])和String(char a[]),int offset,int length,分别用数组a中的全部字符和部分字符创建string对象。 String类也提供将string对象的字符序列存放到数组中的方法: Public void getChars(int start,int.end,char c [],i

    2024年02月11日
    浏览(36)
  • 学习JAVA打卡第四十九天

    Random类 尽管可以使用math类调用static方法random()返回一个0~1之间的随机数。(包括0.0但不包括0.1),即随机数的取值范围是[0.0,1.0]的左闭右开区间。 例如,下列代码得到1~100之间的随机数(包括1和100)  使用Random类的如下构造方法:  创建random对象,其中第一个构造方法

    2024年02月10日
    浏览(30)
  • 学习JAVA打卡第四十五天

    StringBuffer类 StringBuffer对象 String对象的字符序列是不可修改的,也就是说,String对象的字符序列的字符不能被修改、删除,即String对象的实体是不可以再发生变化,例如:对于  StringBuffer有三个构造方法: ⑴StringBuffer(); ⑵StringBuffer(int Size); ⑶StringBuffer(String s);

    2024年02月11日
    浏览(31)
  • 学习JAVA打卡第四十二天

    正则表达式及字符串的替换与分解 ⑴正则表达式 正则表达式是string对象的字符序列,该字符序列中含有具有特殊意义的字符,这些特殊字符称作正则表达式的元字符。  注:由于“   ”代表任何一个字符,所以在正则表达式中如果想使用普通意义的点字符必须使用[.]或5

    2024年02月11日
    浏览(26)
  • 第四十六节 Java 8 Stream

    Java 8 API添加了一个新的抽象称为流Stream,可以让你以一种声明的方式处理数据。 Stream 使用一种类似用 SQL 语句从数据库查询数据的直观方式来提供一种对 Java 集合运算和表达的高阶抽象。 Stream API可以极大提高Java程序员的生产力,让程序员写出高效率、干净、简洁的代码。

    2024年04月23日
    浏览(23)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包