前言
本篇文章主要介绍了 Java 中的 I/O ,其中最重要的是 I/O 流的概念,然后举例说明了一些常用的输入输出操作,包括控制台、文件和网络中的数据输入输出。
概述
I/O 即 Input 和 Output,输入和输出。Java 的核心库 java.io 提供了全面的 I/O 接口。包括:文件读写、标准设备输出等,使其更广泛地应用到文件传输和网络编程中。
Java 中 I/O 是以流为基础进行输入输出的,Java.io 包中的流支持很多种格式,如基本类型、对象、文件等等,所有数据被串行化写入输出流,或者从输入流读入。
流(Stream)
什么是流
Stream 本意为溪流,而 Java 里的 I/O 流里流的不是水,而是数据。数据流是指一组有序的、有起点和重点的字节集合。
以当前的程序为参照,当程序需要读取数据的时候,程序就会与数据源之间建立一个输入流,这个数据源可以是文件,内存,或是网络;当程序需要写入数据的时候,程序就会与目的地之间建立一个输出流。
整个输入输出过程就是数据流入程序再从程序中流出的过程。由于采用了数据流的概念,在程序设计中我们不必关系系统如何实现具体的输入输出,也不必关心输入输出的设备,只需要注意数据流的工作方式。例如,采用键盘写入数据和采用写入文件的方式是一样的。
I/O 流的分类
-
按流向分:
- 输入流:程序可以从中读取数据的流。
- 输出流:程序能向其中写入数据的流。
-
按数据传输单位分:
- 字节流:以字节为单位传输数据的流
- 字符流:以字符为单位传输数据的流
-
按功能分:
- 节点流:用于直接操作目标设备的流
- 过滤流:是对一个已存在的流的链接和封装,通过对数据进行处理为程序提供功能强大、灵活的读写功能。
基本 I/O 流抽象类
JDK 所提供的所有流类位于 java.io 包中,都分别继承自以下四种抽象流类:
InputStream:继承自 InputStream 的流都是用于向程序中输入数据的,且数据单位都是字节(8位)。
OutputStream:继承自 OutputStream 的流都是程序用于向外输出数据的,且数据单位都是字节(8位)。
Reader:继承自 Reader 的流都是用于向程序中输入数据的,且数据单位都是字符(16位)。
Writer:继承自 Writer 的流都是程序用于向外输出数据的,且数据单位都是字符(16位)。
字节流和字符流的区别
-
字节流没有缓冲区,字符流有缓冲区,数据处理后先存入缓存。
-
字节流按字节读写数据,字符流按字符读写数据。
-
字节流可以操作任何类型的数据,字符流只能处理字符类型的数据。
所有的文件在硬盘或在传输时是都是以字节(byte)形式进行的,包括图片、视频等,字符只有到了内存中才会形成,所以在开发中,字节流使用较为广泛。
流的特点
- 先进先出。先写入输出流的数据先被输入流读取到。
- 顺序存取。流中的数据是按读写顺序排序的(RandomAccessFile 除外,使用该流可以从文件的任意位置进行读写操作)。
- 只读或只写。每一种流只能是输入流或输出流的一种,不能同时可读和可写。
结构图
按流分类
按使用场景分类
Scanner 类
java.util.Scanner 是 Java 5 的新特征,我们可以通过 Scanner 类来获取用户的输入。
下面是创建 Scanner 对象的基本语法:
Scanner sc = new Scanner(System.in);
这里的 System.in
是 Java 预置的输入流对象,在 java.lang.System 类中,这样的 I/O 流对象一共有3个:
I/O 流对象 | 说明 |
---|---|
System.in | 标准输入流,通常应用于键盘输入、由主机环境或用户指定的其他输入源 |
System.out | 标准输出流,通常应用于显示输出、由主机环境或用户指定的其他输出目的地 |
System.err | 标准错误输出流,用于显示错误消息或其他应立即引起用户注意的信息 |
Scanner 类的方法有很多,下表中是一些常用的方法:
常用方法 | 描述 |
---|---|
boolean hasNext() | 判断是否还有输入 |
boolean hasNextXxx() | 判断是否还有指定类型的输入 |
boolean hasNextLine() | 判断是否还有下一行输入 |
String next() | 返回下一个分隔符前的字符串 |
xxx nextXxx() | 返回下一个分隔符前的指定类型值 |
String nextLine() | 返回下一行输入 |
Scanner useDelimiter(String pattern) | 设置分隔符(默认分隔符为空格) |
next() 和 nextline() 的区别:
next() 只有在读到有效字符后才能结束输入,有效字符前的空格会被自动去掉,有效字符后的空格被作为分隔符,因此 next() 不能得到带有空格的字符串。
nextLine() 以 Enter 为结束符,也就是说 nextLine() 方法返回的是输入回车之前的所有字符,因此可以获得带空格的字符串。
hasNext() 和 hasNextLine() 的区别:
hasNext() 方法会判断接下来是否有非空字符,如果有,则返回 true,否则返回 false 。
hasNextLine() 方法会根据行匹配模式去判断接下来是否有一行(包括空行),如果有,则返回 true,否则返回 false 。
常用 I/O 操作
控制台 I/O
控制台输入输出一个最重要的应用场景就是笔试了,笔试算法题一般有两种形式(不同的公司不一样):
一种是以力扣为代表的核心代码模式,我们不需要自己构造输入输出,代码框中预设代码已经指定好类名、方法名、参数名,我们只需要关心算法的具体逻辑,将其实现直接返回值即可;
另一种是以 OJ 平台为代表的 ACM 模式,编程过程中的输入输出处理方式需要参考题目输入输出描述,自己进行构造。
对于经常在力扣上刷题的同学来说,习惯了核心代码模式,突然让自己处理输入输出往往会很不适应,甚至出错,因此掌握控制台输入输出十分必要。下面举几个常见的例子:
例1:
输入描述:每个测试用例第一行包含一个正整数n,第二行包含n个正整数nums[i]
示例:
5
1 3 1 1 4
public class Main