Java I/O (Input/Output)——文件字节流

  • 博客主页:誓则盟约
  • 系列专栏:Java SE 专栏
  • 关注博主,后期持续更新系列文章
  • 如果有错误感谢请大家批评指出,及时修改
  • 感谢大家点赞👍收藏⭐评论✍ 

Java I/O 简介

        Java I/O(输入/输出)是 Java 程序中用于处理数据输入和输出的重要部分。

输入流(Input Streams):用于从数据源读取数据。常见的输入流包括FileInputStream(从文件读取)、BufferedInputStream(提高读取效率)等。

输出流(Output Streams):用于将数据写入到目的地。例如FileOutputStream(向文件写入)、BufferedOutputStream(提高写入效率)。

字符流(Reader 和 Writer):处理字符数据,更适合处理文本。如FileReaderFileWriter

缓冲流(Buffered Streams):通过缓冲区来减少实际的 I/O 操作次数,提高性能。

对象流(Object Streams):用于实现对象的序列化和反序列化,如ObjectInputStreamObjectOutputStream

        在实际编程中,根据具体的需求选择合适的 I/O 流可以提高程序的效率和可读性。


计算机总线结构:   

        那么为什么会有I/O呢?其实I/O无时无刻不在我们身边,比如读取硬盘上的文件,网络文件的传输,鼠标键盘输入,也可以是接受单片机发回的数据,而能够支持这些操作的设备就是I/O设备。

我们可以大致看一下整个计算机的总线结构

        最核心的是CPU,CPU像计算机的大脑一样,是计算机的核心部件,几乎所有的计算都是靠这个CPU来进行的,CPU懂的比较多,它可以对各种类型进行计算,但是随着时代的发展对图形的要求越来越高,CPU就略显乏力;于是就出现了GPU(显卡),显卡就是专门对于图形进行计算。

        通过北桥芯片连接到内存,这样CPU就可以对内存进行操作;南桥芯片是用于读取U盘或者硬盘内的数据 。

        常见的I/O设备一般是鼠标、键盘这类通过USB进行传输的外设或者是通过Sata接口或是M.2连接的硬盘。一般情况下,这些设备是由CPU发出指令通过南桥芯片间接进行控制,而不是由CPU直接操作。

        而我们在程序中,想要读取这些外部连接的!O设备中的内容,就需要将数据传输到内存中。而需要实现这样的操作,单单凭借一个小的程序是无法做到的,而操作系统(如:Windows/inux/MacOS)就是专门用于控制和管理计算机硬件和软件资源的软件,我们需要读取一个IO设备的内容时,就可以向操作系统发出请求,由操作系统帮助我们来和底层的硬件交互以完成我们的读取/写入请求。

        JDK提供了一套用于IO操作的框架,为了方便我们开发者使用,就定义了一个像水流一样,根据流的传输方向和读取单位,分为字节流InputStream和OutputStream以及字符流Reader和Writer的IO框架,当然,这里的流指的是数据流,通过流,我们就可以一直从流中读取数据,直到读取到尽头,或是不断向其中写入数据,直到我们写入完成,而这类IO就是我们所说的BIO。


文件字节流:

        字节流一次读取一个字节,也就是一个 byte 的大小,而字符流顾名思义,就是一次读取一个字符,也就是一个 char 的大小(在读取纯文本文件的时候更加适合)。

文件输入流:

在 Java 中,文件输入流FileInputStream用于从文件中读取数据。FileInputStream 允许程序以字节为单位读取文件的内容。

创建方式
通常通过指定要读取的文件路径来创建文件输入流对象。例如:

try {FileInputStream fis = new FileInputStream("your_file_path");// 后续的读取操作
} catch (FileNotFoundException e) {e.printStackTrace();
}

        但是这种方式需要处理各种可能的异常,比如 FileNotFoundException 异常和 IOException 异常,并且需要手动关闭文件,完整代码如下:

public class Hello_World {public static void main(String[] args) { // 想读取一个文件  创建一个文件输入流   使用完把流关闭掉 释放掉 closeFileInputStream stream = null;try {stream = new FileInputStream("绝对路径/相对路径");
//            stream.close();} catch (FileNotFoundException e) {throw new RuntimeException(e);} finally {if (stream != null) {try {stream.close();} catch (IOException e) {throw new RuntimeException(e);}}}}
}

        仅仅是取得文件就如此费劲,不合乎常理。所以有了try-with-resources 语句这种简便方式,try-with-resources 语句是一种用于更方便、更安全地管理资源(如输入流、输出流、数据库连接等)的机制。

优点

  1. 自动资源管理:无需显式地调用 close 方法来关闭资源,避免了因忘记关闭资源而导致的资源泄漏问题。
  2. 简洁的代码:减少了样板代码,使代码更简洁、更易读。

语法格式

try (Resource res = new Resource()) {// 使用资源的操作
} catch (Exception e) {// 异常处理
}

        对于以上示例的完整代码转成try-with-resources 语句如下:

public class Hello_World {public static void main(String[] args) {try(FileInputStream inputStream = new FileInputStream("路径")){ // 直接在try()中定义要在完成之后释放的资源} catch (IOException e){ // 这里变成IOException是因为调用close()可能会出现,而FileNotFoundException是继承自IOException的e.printStackTrace();}// 无需再编写finally语句块,因为在最后自动帮我们调用了close()。}
}

        由此可见,try-with-resources 语句极大地提高了资源管理的便利性和可靠性,使代码更加健壮和易于维护。

数据的传递:

        如图所示,在计算机数据由文件向内存进行传递的形式是以二进制01串进行的,一次一个字节,就像水流一样源源不断的传输,直至文件传输结束。

        数据不断传输过来,那我们如何去读取数据呢?

        调用read()方法是必要的,但是read()方法的调用方式也有很多种,这里主要列出来常见的三种。

1.直接读取

       try(FileInputStream inputStream = new FileInputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt")){ // 直接在try()中定义要在完成之后释放的资源int i = inputStream.read();System.out.println((char)i);int x = inputStream.read();  // 当没有内容后,会返回-1System.out.println((char)x);} catch (IOException e){ // 这里变成IOException是因为调用close()可能会出现,而FileNotFoundException是继承自IOException的e.printStackTrace();}// 无需再编写finally语句块,因为在最后自动帮我们调用了close()。

        由于读取数据返回的是int类型的一个数据,所以我们用int i 去接收它,然后利用强制类型转换把i 转为char类型进行输出。调用一次读取一个字符,当读取完之后会返回-1.这样效率较为低下,所以有下面第二种读取方法。

2.循环读取

         由于读取完之后会返回数字-1,所以可以利用这一性质进行while循环进行读取,直到返回-1时结束循环,代码如下:

try(FileInputStream inputStream = new FileInputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt")){ // 直接在try()中定义要在完成之后释放的资源int i;while ((i = inputStream.read()) != -1) {System.out.print((char)i);}} catch (IOException e){ // 这里变成IOException是因为调用close()可能会出现,而FileNotFoundException是继承自IOException的e.printStackTrace();
}// 无需再编写finally语句块,因为在最后自动帮我们调用了close()。

        通过这种方式就可以一次性对文件内的内容全部读取。但是由于不够灵活,可变性较差,所以还可以用下面第三种方法进行读取。

3.区间读取

        区间读取,顾名思义就是定义一个固定长度的区间,将文件内的内容按照这个区间大小进行读取,当文件未读内容小于区间长度时会以小于区间长度的形式进行最后一次读取,若没有元素可读取时,一样会返回-1。具体代码如下:

        try(FileInputStream inputStream = new FileInputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt")){ // 直接在try()中定义要在完成之后释放的资源System.out.println(inputStream.available()); // 获取有多少个数据可读byte [] bytes = new byte[inputStream.available()]; // 一次读x个数据while (inputStream.read(bytes) != -1)   // 当最后不足x个或者已经没有时,会返回少于x个的数据或者-1System.out.println(new String(bytes));} catch (IOException e){ // 这里变成IOException是因为调用close()可能会出现,而FileNotFoundException是继承自IOException的e.printStackTrace();}// 无需再编写finally语句块,因为在最后自动帮我们调用了close()。

        读取过程中可使用available()方法查询可读数量,在上面的案例中,我将区间长度x设置为了可读长度,这样也可以一次性读取完文件内数据。也可以设置其他int类型的x作为长度参数。

        这种方法在文件输出流常用,一个字节一个字节的读取出来并一个字节一个字节的写入另一个文件,相当于文件的拷贝操作。

       跳过操作:skip()方法。给skip(x)传人参数x,可以设置跳过前几个字节进行读取其下一个字节。


文件输出流:

        文件输出流(FileOutputStream)用于将数据写入到文件中。文件输出流允许您以字节为单位向文件写入数据。

        在写入之前您需要提供要写入的文件的路径和名称。如果文件不存在,它将被创建;如果文件已存在,默认情况下,新写入的数据会覆盖原有的内容。

   try {FileOutputStream fos = new FileOutputStream("your_file.txt");} catch (IOException e) {e.printStackTrace();}

  stream.flush()方法的主要作用是将输出流缓冲区中的数据强制刷新并输出。通常,当我们使用输出流(如 FileOutputStream 、BufferedOutputStream 等)写入数据时,数据并不是立即被发送到目的地(如文件),而是先被存储在缓冲区中。缓冲区的目的是减少实际的 I/O 操作次数,从而提高性能。

        然而,在某些情况下,我们希望确保数据能够立即被发送出去,而不是等到缓冲区填满或者输出流被关闭。这时就可以使用 flush 方法。  

        默认情况下(append的参数默认是false),写入的内容会直接取代原文件内的内容,即覆盖掉。代码如下:

public class Hello_World {public static void main(String[] args) {try(FileOutputStream stream = new FileOutputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt")){stream.write("Hello World".getBytes());  // 直接取代原内容stream.flush();}catch (IOException e){e.printStackTrace();}}
}

        如果想接着文件的内容往后继续写(追加模式),那么只需要把append的参数改为true即可,代码如下:

public class Hello_World {public static void main(String[] args) {try(FileOutputStream stream = new FileOutputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt",true)){ // 加上true 变成追加模式stream.write("Hello World".getBytes());  // 直接取代原内容stream.flush();}catch (IOException e){e.printStackTrace();}}
}

        至此,我们就完成了输出流操作,那么,就可以结合输入流和输出流进行拷贝操作了。

文件的拷贝:

        文件拷贝是将一个文件的内容完整地复制到另一个文件的操作。相关的类有:

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

        文件的拷贝操作一般使用读取数据的第三种方法,区间读取。因为这种方法可以设置足够大的区间,读取速度较快,不需要一个字节一个字节的去读取。下面是一个拷贝的示例代码:

public class Hello_World {public static void main(String[] args) {try(FileInputStream in = new FileInputStream("C:\\Users\\Xxy63\\Desktop\\无限弹窗代码.txt");FileOutputStream out = new FileOutputStream("C:\\Users\\Xxy63\\Desktop\\copy.txt")){byte[] bytes = new byte[1024];int len;while ((len = in.read(bytes)) != -1) {out.write(bytes, 0, len);} // 拷贝速度大大提升}catch (IOException  e){e.printStackTrace();}}
}

        在上述代码中,通过创建输入流 FileInputStream 从源文件读取数据,创建输出流 FileOutputStream 向目标文件写入数据。使用一个缓冲区来提高拷贝效率,每次读取一定数量的字节到缓冲区,然后将缓冲区中的数据写入目标文件,直到读取完源文件的所有内容。

文件拷贝在很多场景中都很有用,比如:

  • 数据备份:将重要文件复制一份以防止数据丢失。
  • 共享文件:将文件拷贝到多个位置以便不同的程序或用户使用。

        例如,如果您有一个包含重要配置信息的文件,为了安全起见,可以定期进行备份拷贝。又或者在一个文件处理系统中,需要将原始文件拷贝到多个不同的目录下以供不同的模块处理。


 “且将新火试新茶,诗酒趁年华。”——《望江南·超然台作》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/393020.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

[C++] 模板进阶:特化与编译链接全解析

文章目录 非类型模板类型形参非类型模板参数代码示例 **模板的特化**为什么要有模板的特化函数模板特化使用场景与示例函数模板特化的实现细节 类模板特化全特化示例 偏特化部分优化通过进一步限制模板参数进行特化偏特化为指针类型示例:偏特化为引用类型示例&#…

menuconfig+Kconfig的简单配置

目录 1.背景 2.管理方案 2.1:.h中直接定义 2.2:.batCmake 2.3:Kconfig 2.3.1 环境安装 2.3.2 代码 2.3.2.1 目录结构 2.3.2.2 ble目录下的Kconfig 2.3.2.3 hardware目录下的Kconfig 2.3.2.4 rtos目录下的Kconfig 2.3.2.5 根目录 …

申请专利需要准备哪些材料?

申请专利需要准备哪些材料?

实践致知第17享:电脑忽然黑屏的常见原因及处理方法

一、背景需求 小姑电话说:最近,电脑忽然就黑屏了(如下图所示),但是等待几十秒甚至一分钟,电脑就能自然恢复了,这种状况一天能出现三四次,怎么办? 二、分析诊断 电脑黑屏…

keeplive配置详解与haproxy配置详解

一、keepalive相关知识 1.1 keepalive介绍 keepalive即LVS集群当中的高可用架构,只是针对调度器的高可用。是高可用的HA架构。 keepalive就是基于VRRP协议来实现LVS高可用的方案。 1、组播地址 224.0.0.18,根据组播地址进行通信,主备之间发…

Java多线程-----定时器(Timer)及其实现

目录 一.定时器简介: 二.定时器的构造方法与常见方法: 三.定时器的模拟实现: 思路分析: 代码实现: 在开发中,我们经常需要一些周期性的操作,例如每隔几分钟就进行某一项操作,这…

标准IO——文件定位、文件IO

续:feof、ferror(检测一个流是否出错)、clearerr(清除一个流出错的标记)。 一、标准IO文件定位 1、fseek(定位) int fseek(FILE *stream , long offset(偏移长度) , int whence(偏移起始位置)) 其中when…

阿里云SMS服务C++ SDK编译及调试关键点记录

一. 阿里云SMS服务开通及准备工作 在阿里云官网上完成这部分的工作 1. 申请资质 个人or企业 我这里是用的企业资质 2. 申请签名 企业资质认证成功后,会自动赠送一个用于测试的短信签名 也可以自己再进行申请,需要等待审核。 3. 申请短信模板 企…

还没用过OBS Studio?快来提升你的技术分享效率!

前言 在浩瀚的数字海洋中,有这么一款神器,它低调却光芒四射,默默改变着无数内容创作者的命运;嘿,你猜怎么着?它既不是天价的专业设备,也不是遥不可及的神秘黑科技,而是开源世界的瑰宝…

本地Gitlab-runner自动编译BES项目

0 Preface/Foreword 1 Gitlab-runner配置情况 具体情况如下: Gitlab-ruuner运行在wsl 1中的Ubuntu 18.04 distro上专门为GitLab-runner分配了一个用户,名为gitlab-runner 2 自动编译 2.1 找不到编译工具链 根据错误提示,交叉编译工具链未找…

深入理解接口测试:实用指南与最佳实践(四)IHRM管理系统实战-项目分析

​ ​ 您好,我是程序员小羊! 前言 这一阶段是接口测试的学习,我们接下来的讲解都是使用Postman这款工具,当然呢Postman是现在一款非常流行的接口调试工具,它使用简单,而且功能也很强大。不仅测试人员会使用…

牛!手机上轻松部署大模型全攻略!

当前AI革命中,大模型发挥关键角色,其理论基础在于Scaling Law。简单来说就是,随着数据、参数和计算能力的提升,模型能力增强,展现出小规模模型所不具备的“涌现能力”。众多AI企业推出开源大模型,规模按扩展…

红黑树的概念和模拟实现[C++]

文章目录 红黑树的概念一、红黑树的性质红黑树原理二、红黑树的优势和比较 红黑树的模拟实现构建红黑树的数据结构定义节点的基本结构和初始化方式插入新节点插入新节点的颜色调整颜色和结构以满足红黑树性质 红黑树的应用场景 红黑树的概念 一、红黑树的性质 红黑树是一种自平…

Redis系列之Redis Sentinel

概述 Redis主从集群,一主多从模式,包括一个Master节点和多个Slave节点。Master负责数据的读写,Slave节点负责数据的查询。Master上收到的数据变更,会同步到Slave节点上实现数据的同步。通过这种架构实现可以Redis的读写分离&…

工具|阅读PDF时鼠标显示为小手中有向下箭头解决方法

由于工作中,会大量阅读PDF文档,如手册,规格书,各种图纸等,因此好用的PDF工具必不可少。我主要习惯用福昕阅读器,标注比较方便。 所以,本文主要以福昕阅读器为主,当然也适用于其他的阅…

Docker Volume(存储卷)

一、认识 1.1 概念 存储卷就是将宿主机的本地文件系统中存在的某个目录直接与容器内部的文件系统上的某一目录建立绑定关系。这意味着,在容器中的这个目录下写入数据时,容器会将内容直接写入到宿主机上与此容器建立了绑定关系的目录 在宿主机上的这个…

实验8-1-6 在数组中查找指定元素

本题要求实现一个在数组中查找指定元素的简单函数。 函数接口定义: int search( int list[], int n, int x );其中list[]是用户传入的数组;n(≥0)是list[]中元素的个数;x是待查找的元素。如果找到 则函数search返回…

基于Golang实现Kubernetes边车模式

本文介绍了如何基于 Go 语言实现 Kubernetes Sidecar 模式,并通过实际示例演示创建 Golang 实现的微服务服务、Docker 容器化以及在 Kubernetes 上的部署和管理。原文: Sidecar Pattern with Kubernetes and Go[1] 在这篇文章中,我们会介绍 Sidecar 模式…

软件测试学习笔记

测试学习 1. 测试流程2. Bug的提出什么是bugbug 的描述bug 级别 3. 测试用例的设计什么是测试用例测试用例应如何设计基于需求的设计方法等价类边界值场景法正交表法判定表法错误猜测法 4. 自动化测试回归测试自动化分类 5. 安装 webdriver-manager 和 selenium第一个web自动化…

链表List

简介 STL中的List与顺序表vector类似,同样是一种序列式容器,其原型是带头节点的双向循环链表。 List的使用 list中的接口比较多,此处类似,只需要掌握如何正确的使用,然后再去深入研究背后的原理,已达到可…