java IO操作

article/2025/10/5 14:16:36

系列文章目录

第一章 万事万物皆对象
第二章 java构造器
第三章 java访问权限
第四章 java初始化
第五章 java继承
第六章 java重载与重写
第七章 java接口和抽象类
第八章 java数组
第九章 java内部类
第十章 java时间操作
第十一章 final关键字
第十二章 java字符串
第十三章 java异常
第十四章 java泛型
第十五章 java IO操作
第十六章 java NIO操作
第十七章 java zip压缩
第十八章 java反射
第十九章 java反射之Type接口
第二十章 java8流式操作
第二十一章 java8函数式编程


文章目录

    • 普通IO
      • 字节流
        • 输入流
          • 继承关系
          • 方法介绍
            • 读取数据
            • 关闭流
            • 跳过指定字节
            • 标记与重置
            • 可读字节数
        • 输出流
          • 方法介绍
            • 写入数据
            • 关闭流
            • 刷新
      • 字符流
      • RandomAccessFile类
      • 常用的IO操作
        • 缓冲输入文件
        • 读取字符
        • 格式化数据
        • 文件输出
    • 标准IO
      • 标准输入
      • 重定向标准I/O
      • 序列化流和反序列化流
        • 序列化接口
        • ObjectOutputStream序列化流
        • ObjectInputStream反序列化流


原文链接 https://zhhll.icu/2020/java基础/IO/1.java基础之IO操作/

用户进程发起请求,内核接收到请求后,从I/O设备中获取数据到buffer中,再将buffer中的数据copy到用户进程的地址空间,该用户进程获取到数据后再响应客户端。

数据输入到buffer需要时间,从buffer复制数据至进程也需要时间,根据在这两段时间内等待方式不同,I/O动作可分为五种模式

  • 阻塞I/O(Blocking I/O)
  • 非阻塞I/O(Non-Blocking I/O)
  • I/O复用(I/O Multiplexing)
  • 信号驱动I/O
  • 异步I/O

java的I/O操作在类的java.io包中

  • 基于字节操作的I/O接口: InputStream和OutputStream

  • 基于字符操作的I/O接口: Writer和Reader

  • 基于磁盘操作的I/O接口: File

  • 基于网络操作的I/O接口: Socket(网络编程,不在io包中)

普通IO

  1. 字节流对应原生的二进制数据

  2. 字符流对应字符数据,会自动处理与本地字符集之间的转换

  3. 缓冲流可以提高性能,通过减少底层API的调用次数来优化IO

字节流

读取文件是输入流,写文件是输出流(输入输出是对于程序而言的)

输入流

在这里插入图片描述

继承关系

字节输入流都继承自InputStream,InputStream表示从不同数据源产生输入的类,这些数据源包括

  1. 字节数组
  2. String对象
  3. 文件
  4. 管道,从一端输入,从另一端输出
  5. 一个由其他种类的流组成的序列,然后把它们汇聚为一个流
  6. 其他数据源,如Internet连接
功能构造器如何使用
ByteArrayInputStream允许将内存的缓冲区当做InputStream使用缓冲区,字节将从中取出将其与FilterInputStream对象相连以提供有用接口
StringBufferInputStream(已废除)将String转换为InputStream字符串。底层实现实际使用StringBuffer将其与FilterInputStream对象相连以提供有用接口
FileInputStream用于从文件中读取信息字符串,表示文件名、文件或FileDescriptor对象
PipedInputStream产生用于写入相关PipedOutputStream的数据。实现管道化概念PipedOutputStream作为多线程中的数据源
SequenceInputStream将两个或多个InputStream对象转换成一个InputStream两个InputStream对象或一个容纳InputStream对象的容器Enumeration
FilterInputStream抽象类,作为装饰器的接口,为其他的InputStream类提供有用的功能

FilterInputStream类型子类包括以下几种

功能构造器如何使用
DataInputStream与DataOutputStream搭配使用,按照移植方式从流读取基本数据类型InputStream包含用于读取基本数据类型的全部接口
BufferedInputStream使用它可以防止每次读取时都得进行实际写操作,提供了缓冲区的操作,提高IO的性能InputStream本质上不提供接口,只是向进程添加缓冲功能
LineNumberInputStream(已废除)跟踪输入流的行号,可调用getLineNumber()和setLineNumber(int)InputStream,可以指定缓冲区大小仅增加了行号
PushbackInputStream具有能弹出一个字节的缓冲区,因此可以将读到的最后一个字符回退InputStream通常作为编译器的扫描器
方法介绍
读取数据
// 每次读取一个字节
public abstract int read() throws IOException
// 使用字节数组作为缓冲区,将读到的字节填充至缓冲区中
public int read(byte b[]) throws IOException
// 将读到的字节填充至字节数组的指定位置上
// off 起始位置
// len 长度
public int read(byte b[], int off, int len) throws IOException
关闭流

使用close方法来关闭流,在1.7及之后建议使用try-with-resources语句来使用流,这样可以避免显示的调用close()方法,减少一些重复代码

public void close() throws IOException
跳过指定字节

可以使用skip方法来跳过指定数目的字节,相当于把流中当前读到的位置向后移动若干个字节。

public long skip(long n) throws IOException

注意:该方法可能在向后移动的时候,没有达到指定字节就到达了流的末尾,所以并不一定会跳过指定的字节,该方法返回值为实际跳过的字节数

标记与重置

标记与重置方法一般联合使用,以实现流中部分内容可重复读取

// mark方法在当前读取位置进行标记
// readlimit表示允许重复读取的字节,只能从标记的位置再次重复向后读取所指定的字节
public synchronized void mark(int readlimit)// reset方法将流的当前读取位置移动到上次标记的位置
public synchronized void reset() throws IOException// 不是所有的流都支持标记功能,需要使用该方法来判断当前流是否支持标记功能
public boolean markSupported()
可读字节数

当read方法被调用时,如果当前流中没有可用的数据,该操作就会被阻塞,available()方法就是返回当前流中还有多少字节可以读取

public int available() throws IOException

输出流

在这里插入图片描述

字节输入流都继承自OutputStream,该类决定了输出所要去的目标,字节数组、文件或管道

功能构造器如何使用
ByteArrayOutputStream在内存中创建缓冲区。所有送往流的数据都要放置在此缓冲区缓冲区初始大小用于指定数据的目的地
FileOutputStream用于将信息写入文件字符串,表示文件名、文件或FileDescriptor对象
PipedOutputStream任何写入其中的信息都会自动作为相关PipedInputStream的输出,实现管道化概念PipedInputStream指定用于多线程的数据的目的地
FilterOutputStream抽象类,作为装饰器的接口,为其他OutputStream提供有用的功能

FilterOutputStream类型子类包括

功能构造器如何使用
DataOutputStream与DataInputStream搭配使用,可以按照移植方式向流中写入基本数据类型OutputStream包含用于写入基本数据类型的全部接口
PrintStream用于产生改格式化输出。其中DataOutputStream处理数据的存储,PrintStream处理显示OutputStream,可以用boolean值指示是否每次换行时清空缓冲区应该是对OutputStream对象的final封装
BufferedOutputStream使用它以避免每次发送数据时都进行实际的写操作,可以调用flush()清空缓冲区OutputStream,可以指定缓冲区大小只是向进程添加缓冲功能
方法介绍
写入数据
// 每次写入一个字节
public abstract void write(int b) throws IOException
// 使用字节数组作为缓冲区,写入整个字节数组的内容
public void write(byte b[]) throws IOException
// 写入字节数组的部分内容
// off 起始位置
// len 长度
public void write(byte b[], int off, int len) throws IOException
关闭流

使用close方法来关闭流,在1.7及之后建议使用try-with-resources语句来使用流,这样可以避免显示的调用close()方法,减少一些重复代码

public void close() throws IOException
刷新

flush()方法用来强制要求OutputStream对象将暂存于内部缓冲区的数据立即进行实际的写入(一般情况下不需要手动的调用该方法,在内部缓冲区填充满了之后,会自动执行实际的写入操作,在调用close()方法时也会自动调用flush()方法)

有些OutputStream类中维护内部缓冲区是为了减少实际的写入操作次数,来提升性能

public void flush() throws IOException

字符流

在这里插入图片描述

InputStream和OutputStream是面向字节I/O的,而Reader和Writer则提供兼容Unicode和面向字符I/O的功能,InputStreamReader可以把InputStream转换为Reader,而OutputStreamWriter可以把OutputStream转换为Writer

RandomAccessFile类

适用于由大小已知的记录组成的文件,可以使用seek()将文件指针从一条记录移动到另一条记录,然后对记录进行读取和修改。文件中记录的大小不一定相同,只要我们能确定那些记录有多大以及它们在文件中的位置即可。

RandomAccessFile类不是InputStream或者OutputStream的子类,只是实现了DataInput和DataOutput接口,没有使用InputStream和OutputStream的任何功能,所有方法都是独立的,大部分为native方法

既可以读文件,又可以写文件,可以进行文件的随机读写

// 有两种模式,"rw"读写 和 "r"只读
RandomAccessFile faf = new RandomAccessFile(file,"rw");
// 文件指针,打开文件时指针在开头pointer=0// 写操作,只写一个字节,同时指针指向下一个位置,准备再次写入
raf.write(int)// 读文件时要把指针移到头部
raf.seek(0);
// 读操作,读一个字节
int b = raf.read()

常用的IO操作

缓冲输入文件

想要打开一个文件进行字符输入,可以使用FileReader对象,然后传入一个String或者File对象作为文件名。为了提高速度,对文件进行缓冲,可以将所产生的引用传递给一个BufferedReader构造器。BufferedReader提供了lines()方法,会产生一个Stream对象

 public static void read(String file) {try(BufferedReader bufferedReader = new BufferedReader(new FileReader(file))){String line = null;while((line = bufferedReader.readLine()) != null){System.out.println(line);}} catch (IOException e) {throw new RuntimeException("读取失败",e);}}

读取字符

public static void read() throws IOException {StringReader stringReader = new StringReader("qaw试试");int c;while ((c = stringReader.read()) != -1) {System.out.println((char)c);}}

StringReader的read方法是以int形式返回的下一个字节,所以打印的时候类型必须转为char

格式化数据

要读取格式化数据,可以使用DataInputStream,是面向字节的,所以要使用InputStream类

文件输出

FileWrite对象用于向文件写入数据,通常使用BufferedWriter将其包装起来增加缓冲的功能,

try(BufferedReader in = new BufferedReader(new StringReader("1111111111111111\n2222222222222\n3333333333"));PrintWriter printWriter = new PrintWriter(new BufferedWriter(new FileWriter("test.txt")))){in.lines().forEach(printWriter :: println);} catch (IOException e) {e.printStackTrace();}

标准IO

标准输入流Systom.in、标准输出流System.out、标准错误流Sytem.err。

System.out和System.err是预先包装成了PrintStream对象,但是System.in没有进行包装,属于原生的InputStream,所以在读取时需要对其进行包装

标准输入

通常一行一行地读取输入,将System.in包装成BufferedReader

public static void main(String[] args) {try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in))) {String line;while ((line = bufferedReader.readLine()) != null) {if (line.equals("end")) {break;}System.out.println(line);}} catch (IOException e) {}}

重定向标准I/O

System类提供了简单的静态方法调用,重定向标准输入流、标准输出流和标准错误流

  • setIn(InputStream)

  • setOut(PrintStream)

  • setErr(PrintStream)

public static void main(String[] args) {try(BufferedInputStream bufferedInputStream = new BufferedInputStream(new FileInputStream("C:\\Users\\sinosoft\\Desktop\\剩余工作.txt"));PrintStream printStream = new PrintStream(new BufferedOutputStream(new FileOutputStream("C:\\Users\\sinosoft\\Desktop\\剩余工作副本.txt")))){System.setIn(bufferedInputStream);System.setOut(printStream);try (BufferedReader bufferedReader = new BufferedReader(new InputStreamReader(System.in))) {String line;while ((line = bufferedReader.readLine()) != null) {if (line.equals("end")) {break;}System.out.println(line);}} catch (IOException e1) {}}catch (IOException e){}}

序列化流和反序列化流

对象的序列化就是将对象转化为byte序列,反之叫做反序列化,ObjectOutputStream是序列化流,ObjectInputStream是反序列化流

序列化接口

对象必须实现序列化接口Serializable,才可以进行序列化,该接口是一个标准,如果不想某个字段进行序列化,可以使用transient来修饰字段,使得该字段不进行序列化,可以通过重写writeObject和readObject方法来进行手动序列化

静态变量也不能进行序列化,因为所有的对象都共享同一份静态变量值

ObjectOutputStream序列化流

主要的方法是writeObject,存储对象的类、类的签名以及这个类及其父类中所有非静态和非瞬时的字段的值

public final void writeObject(Object obj) throws IOException {if (enableOverride) {writeObjectOverride(obj);return;}try {writeObject0(obj, false);} catch (IOException ex) {if (depth == 0) {writeFatalException(ex);}throw ex;}
}

ObjectInputStream反序列化流

主要的方法是readObject,读回对象的类、类的签名以及这个类及其父类中所有非静态和非瞬时的字段的值

public final Object readObject()throws IOException, ClassNotFoundException {return readObject(Object.class);
}

在反序列化生成对象时,构造函数并不会执行,因为反序列化是要得到存储时的状态,如果调用构造函数就会生成新的


http://chatgpt.dhexx.cn/article/wA14GEWj.shtml

相关文章

Linux基础 IO操作

一、IO接口的基本认识与使用 在c语言中,我们对文件用fopen,fread,fwrite,fclose等等c语言标准库函数来进行操作,实际上这些库函数是对系统调用接口的一个封装,目的是操作简单,方便编程。 以下…

Java中的IO操作

IO 操作是我以前不太理解的操作。总是觉得很神奇,然后一脸懵逼。今天终于鼓起勇气,认真的学习IO方面的知识。有理解不妥当或者不正确的地方,还请各位大佬指正。 一、java的io发生在哪? 这个问题,我们必须首先明确下。…

基础IO-文件IO操作

目录 1.C语言文件IO相关操作(库函数IO接口) 2. 系统调用IO接口 1.C语言文件IO相关操作(库函数IO接口) fopen--打开文件,返回一个文件的操作句柄,有了句柄才能对指定的文件进行操作。 FILE *fopen(char *p…

Java文件IO操作

目录 一、了解什么是文件 狭义的文件: 广义的文件: 二、文件的路径 ①文件的绝对路径 ②文件的相对路径 三、Java对于文件的操作 File类的构造方法 File类的普通方法 四、对于文件的内容操作 ①FileInputStream(文件输入流&#xf…

IO操作

一、简介 IO流是用来处理设备之间的数据传输的,数据一般储存在设备上 设备:硬盘,光盘,内存等等,内存是临时储存的 Java对数据的操作是采用流的方式 按照操作的数据分为:字节流和字符流 暗照流向&#x…

Java基础--IO操作

一、IO原理及分类 一、IO原理 1、I/O是Input/Output的缩写,I/O技术是非常实用的技术,用于处理设备之间的数据传输,如读写文件,网络通信等。 2、java程序中对于数据的输入/输出操作一般都是以流的方式进行 3、java.io包下提供各…

关于I/O操作

https://blog.csdn.net/deram_boy/article/details/51191425 提及这个,就想到了各种文件操作,这种流,什么reader 啊, writer啊,buffer啊,file啊。但是综合的东西与总结,只是曾经了解。因为工作…

IO操作概念

IO:输入输出,对程序的输入与输出 文件:一组有关数据的集合 文件名:这组数据的名字 linux系统中所有文件均属于文件(通过文件系统进行管理) 按类型分类: 1、普通文件 (…

I/O操作

前言 I/O操作即操作存储器,所以想了解I/O操作,需要了解一点存储器相关知识点。 计算机存储器 存储器是计算机的核心部件之一,在完全理想的状态下,存储器应该要同时具备以下三种特性: (1)速度…

操作系统_04_IO相关(个人总结)

声明: 1. 本文为我的个人复习总结, 并非那种从零基础开始普及知识 内容详细全面, 言辞官方的文章 2. 由于是个人总结, 所以用最精简的话语来写文章 3. 若有错误不当之处, 请指出 软硬链接 软链接(快捷方式): 删除原文件时, 软链接则失效 硬链接…

gcc 命令

gcc 命令 GCC 编译器 gcc 命令格式 gcc 选项 文件名字主要选项 -c&#xff1a; 只编译不链接为可执行文件&#xff0c;编译器将输入的.c 文件编译为.o 的目标文件。-o&#xff1a; <输出文件名>用来指定编译结束以后的输出文件名&#xff0c;如果不使用这个选项的话 G…

Ubuntu Linux gcc的常用命令

目录 一、gcc简介 二、简单的gcc编译 1 预处理 2 编译为汇编语言代码 3 汇编 4 连接 三、多个程序文件的编译 四、检查错误 五、连接库文件 1 编译成可执行文件 2 链接 3 强制链接时使用静态链接库 六、总结 一、gcc简介 Linux系统下的Gcc&#xff08;GNU C Compi…

Linux GCC 编译详解

文章目录 一、GCC 编译器简介二、GCC 工作流编程语言的发展GCC 工作流程gcc 和 g 的区别 三、使用 GCC 编译GCC 编译格式GCC 编译流程多个源文件编译 一、GCC 编译器简介 首先&#xff0c;什么是编译器呢&#xff1f; 我们可以使用编辑器&#xff08;如 linux 下的 vi、window…

gcc编译命令的常用选项——强烈推荐大家使用 -Wall 选项

C程序编编译的过程分为如下四个阶段 1.预处理&#xff1a;头文件展开&#xff08;#include&#xff09;、宏替换&#xff08;#define&#xff09;、条件编译&#xff08;#ifdef&#xff09;(.i)使用预处理器&#xff08;预处理阶段处理的都是以#开头的代码&#xff09; 2.编译…

Linux(gcc编译原理、过程以及常用调试命令)

PS&#xff1a;红色字体表示重点&#xff0c;绿色字体表示重要标题&#xff0c;块引用中表示Linux终端中的命令。 1.gcc / g的安装 命令&#xff1a;sudo apt install gcc /gcc 2.gcc编译连接 //main.c文件 &#xff08;1&#xff09;预编译 ①删除所有的“#define”&#…

Linux中gcc编译步骤

我们这里以C语言为例&#xff0c;看看C语言程序在Linux中编译执行的详细步骤 1.创建一个.c文件 2.写入一些简单的代码 我们使用gcc -E filename.c -o filename.i 命令对程序先进行预处理&#xff1a;处理头文件、宏定义&#xff0c;接下来我们修改一下头文件&#xff0c;使用错…

Linux GCC常用命令

目录 一、示例一 1.简单编译 1.1预处理 1.2编译为汇编代码 1.3汇编 1.4连接 2.多个程序文件的编译 3检错 4库文件连接 二、示例二 1.准备hello.c 2.预处理 3.编译 4.汇编 5.链接 6.分析ELF文件 6.1ELF文件的段 6.2反汇编ELF 一、示例一 1.简单编译 创建文件mkd…

GCC编译过程及基本命令总结

一、GCC简介 GCC即GNU Compiler Collection&#xff0c;原本只是针对C语言的编译工具&#xff0c;现在已经变成了一个工具集&#xff0c;包含了C、C、JAVA等语言的集合体。 管理和维护&#xff1a;由GNU项目负责。 二、GCC对C、C的编译流程 (1) 预处理&#xff08;Preproce…

Linux 下的 gcc 编译常用命令

gcc&#xff1a;Linux下的一款编译器。 gcc工作流程&#xff1a; 完整编译流程&#xff1a; 创建一个c语言源文件&#xff1a; gcc -E hello.c -o hello.i&#xff08;头文件展开&#xff0c;进行源文件中的宏替换&#xff0c;注释过滤&#xff09; gcc -S hello.i -o hello.…

GCC命令编译

1.GCC编译过程&#xff08;原理如下&#xff0c;使用命令请见2&#xff09; 从 hello.c 到 hello(或 a.out)文件&#xff0c; 必须历经 hello.i、 hello.s、 hello.o&#xff0c;最后才得到 hello(或a.out)文件&#xff0c;分别对应着预处理、编译、汇编和链接 4 个步骤&#…