本文最后更新于 2022年9月28日 早上
概述
IO
通过数据流、序列化和文件系统提供系统输入和输出。
流
流是一个很形象的概念,当程序需要读取数据的时候,就会开启一个通向数据源的流,这个数据源可以是文件,内存,或是网络连接。类似的,当程序需要写入数据的时候,就会开启一个通向目的地的流。这时候你就可以想象数据好像在这其中“流”动一样。
原理
Java 把这些不同来源和目标的数据都统一抽象为数据流。Java 语言的输入输出功能是十分强大而灵活的,美中不足的是看上去输入输出的代码并不是很简洁,因为你往往需要包装许多不同的对象。
在 Java 类库中,IO 部分的内容是很庞大的,因为它涉及的领域很广泛:标准输入输出,文件的操作,网络上的数据流,字符串流,对象流,zip 文件流。
File 文件类
在 Java 中,File 类是 java.io 包中唯一代表磁盘文件本身的对象,也就是说,如果希望在程序中操作文件和目录,则都可以通过 File 类来完成。File 类定义了一些方法来操作文件,如新建、删除、重命名文件和目录等。
File 类不能访问文件内容本身,如果需要访问文件内容本身,则需要使用输入/输出流。
File 类提供了如下三种形式构造方法。
- File(String path):如果 path 是实际存在的路径,则该 File 对象表示的是目录;如果 path 是文件名,则该 File 对象表示的是文件。
- File(String path, String name):path 是路径名,name 是文件名。
- File(File dir, String name):dir 是路径对象,name 是文件名。
使用任意一个构造方法都可以创建一个 File 对象,然后调用其提供的方法对文件进行操作。在表中列出了 File 类的常用方法及说明。
方法名称 |
说明 |
boolean canRead() |
测试应用程序是否能从指定的文件中进行读取 |
boolean canWrite() |
测试应用程序是否能写当前文件 |
boolean delete() |
删除当前对象指定的文件 |
boolean exists() |
测试当前 File 是否存在 |
String getAbsolutePath() |
返回由该对象表示的文件的绝对路径名 |
String getName() |
返回表示当前对象的文件名或路径名(如果是路径,则返回最后一级子路径名) |
String getParent() |
返回当前 File 对象所对应目录(最后一级子目录)的父目录名 |
boolean isAbsolute() |
测试当前 File 对象表示的文件是否为一个绝对路径名。该方法消除了不同平台的差异,可以直接判断 file 对象是否为绝对路径。在 UNIX/Linux/BSD 等系统上,如果路径名开头是一条斜线/ ,则表明该 File 对象对应一个绝对路径;在 Windows 等系统上,如果路径开头是盘符,则说明它是一个绝对路径。 |
boolean isDirectory() |
测试当前 File 对象表示的文件是否为一个路径 |
boolean isFile() |
测试当前 File 对象表示的文件是否为一个“普通”文件 |
long lastModified() |
返回当前 File 对象表示的文件最后修改的时间 |
long length() |
返回当前 File 对象表示的文件长度 |
String[] list() |
返回当前 File 对象指定的路径文件列表 |
String[] list(FilenameFilter) |
返回当前 File 对象指定的目录中满足指定过滤器的文件列表 |
boolean mkdir() |
创建一个目录,它的路径名由当前 File 对象指定 |
boolean mkdirs() |
创建一个目录,它的路径名由当前 File 对象指定 |
boolean renameTo(File) |
将当前 File 对象指定的文件更名为给定参数 File 指定的路径名 |
IO 流的分类
流向和数据类型
根据数据的流向分为:输入流 和 输出流。
- 输入流 :把数据从
其他设备
上读取到内存
中的流。
- 输出流 :把数据从
内存
中写出到其他设备
上的流。
根据数据的类型分为:字节流 和 字符流。
- 字节流 :以字节为单位,读写数据的流。
- 字符流 :以字符为单位,读写数据的流。
分类之后对应的父类
|
输入流 |
输出流 |
字节流 |
字节输入流 InputStream |
字节输出流 OutputStream |
字符流 |
字符输入流 Reader |
字符输出流 Writer |
注:
由这四个类的子类名称基本都是以其父类名作为子类名的后缀。
如:InputStream 的子类 FileInputStream。
如:Reader的子 类 FileReader。
具体分类
分类 |
字节输入流 |
字节输出流 |
字符输入流 |
字符输出流 |
抽象基类 |
InputStream |
OutputStream |
Reader |
Writer |
访问文件 |
FileInputStream |
FileOutputStream |
FileReader |
FileWriter |
访问数组 |
ByteArrayInputStream |
ByteArrayOutpuStream |
CharArrayReader |
CharArrayWriter |
访问管道 |
PipedInputStream |
PipedOutputStream |
PipedReader |
PipedWriter |
访问字符串 |
|
|
StringReader |
StringWriter |
缓冲流 |
BufferedInputStream |
BufferedOutputStream |
BufferedReader |
BufferedWriter |
转换流 |
|
|
InputStreamReader |
OutputStreamWriter |
对象流 |
ObjectInputStream |
ObjectOutputStream |
|
|
过滤流 |
FilterInputStream |
FilterOutputStream |
FilterReader |
FilterWriter |
打印流 |
|
PrintStream |
|
PrintWriter |
推回输入流 |
PushbackInputStream |
|
PushbackReader |
|
数据流 |
DataInputStream |
DataOutputStream |
|
|
注:倾斜代表抽象类,无法创建实例,红色表示节点流,必须直接与指定物理节点关联。
代码示例
File 类
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25
| import java.io.File; import java.util.Date;
public class FileTest {
public static void main(String[] args) { String path = "D:/"; File f = new File(path, "test.txt"); System.out.println("D:\\test.txt文件信息如下:"); System.out.println("============================================"); System.out.println("文件长度:" + f.length() + "字节"); System.out.println("文件或者目录:" + (f.isFile() ? "是文件" : "不是文件")); System.out.println("文件或者目录:" + (f.isDirectory() ? "是目录" : "不是目录")); System.out.println("是否可读:" + (f.canRead() ? "可读取" : "不可读取")); System.out.println("是否可写:" + (f.canWrite() ? "可写入" : "不可写入")); System.out.println("是否隐藏:" + (f.isHidden() ? "是隐藏文件" : "不是隐藏文件")); System.out.println("最后修改日期:" + new Date(f.lastModified())); System.out.println("文件名称:" + f.getName()); System.out.println("文件路径:" + f.getPath()); System.out.println("绝对路径:" + f.getAbsolutePath()); } }
|
输出:
D:\test.txt文件信息如下:
============================================
文件长度:0字节
文件或者目录:是文件
文件或者目录:不是目录
是否可读:可读取
是否可写:可写入
是否隐藏:不是隐藏文件
最后修改日期:Tue Sep 27 16:06:10 GMT+08:00 2022
文件名称:test.txt
文件路径:D:\test.txt
绝对路径:D:\test.txt
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20
| import java.io.File; import java.io.IOException;
public class FileTest01 {
public static void main(String[] args) throws IOException { File f = new File("D:\\test.txt"); if (f.exists()) { boolean delete = f.delete(); System.out.println(delete); } boolean newFile = f.createNewFile(); System.out.println(newFile); } }
|
输出:
true
true
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| import java.io.File;
public class FileTest02 {
public static void main(String[] args) { String path = "D:/config/"; File f = new File(path); if (f.exists()) { boolean delete = f.delete(); System.out.println(delete); } boolean mkdir = f.mkdir(); System.out.println(mkdir); } }
|
输出:
true
true
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21
| import java.io.File;
public class FileTest03 {
public static void main(String[] args) { File f = new File("D:/"); System.out.println("文件名称\t\t文件类型\t\t文件大小"); System.out.println("==================================================="); String[] fileList = f.list(); assert fileList != null; for (String s : fileList) { System.out.print(s + "\t\t"); System.out.print((new File("D:/", s)).isFile() ? "文件" + "\t\t" : "文件夹" + "\t\t"); System.out.println((new File("D:/", s)).length() + "字节"); } } }
|
输出:
文件名称 文件类型 文件大小
===================================================
$RECYCLE.BIN 文件夹 0字节
config 文件夹 0字节
data 文件夹 0字节
home 文件夹 0字节
JamWorkspaceAndroid 文件夹 0字节
note 文件夹 4096字节
oracle 文件夹 4096字节
pagefile.sys 文件 8053063680字节
picture 文件夹 12288字节
Program Files 文件夹 4096字节
Program Files (x86) 文件夹 4096字节
project 文件夹 0字节
study 文件夹 4096字节
System Volume Information 文件夹 0字节
test.txt 文件 192字节
test1.txt 文件 0字节
Users 文件夹 0字节
virtual 文件夹 0字节
软件 文件夹 4096字节
字节流
文件的复制(字节流)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| import java.io.*;
public class InputOutputTest {
public static void main(String[] args)throws IOException { File source = new File("D:\\test.txt") ; File target = new File("D:\\test1.txt") ; InputStream inStream = new FileInputStream(source) ; OutputStream outStream = new FileOutputStream(target) ; byte[] byteArr = new byte[1024]; int readSign ; while ((readSign = inStream.read(byteArr)) != -1){ outStream.write(readSign); } outStream.close(); inStream.close(); } }
|
字符流
文件的复制(字符流),其速度效率要快于字节流读取,对于非文本文件(视频文件、音频文件、图片),只能使用字节流!
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23
| import java.io.*;
public class ReaderWriterTest {
public static void main(String[] args) throws Exception { File readerFile = new File("D:\\test.txt") ; File writerFile = new File("D:\\test1.txt") ; Reader reader = new FileReader(readerFile) ; Writer writer = new FileWriter(writerFile); int readSign; while ((readSign = reader.read()) != -1) { writer.write(readSign); } writer.flush(); writer.close(); reader.close(); } }
|
字节缓冲流
文件的复制(字节缓冲流,加速字节读取)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| import java.io.*;
public class BufferedInputOutputTest {
public static void main(String[] args) throws Exception { File source = new File("D:\\test.txt"); File target = new File("D:\\test1.txt"); InputStream bufInStream = new BufferedInputStream(new FileInputStream(source)); OutputStream bufOutStream = new BufferedOutputStream(new FileOutputStream(target)); int readSign; while ((readSign = bufInStream.read()) != -1) { bufOutStream.write(readSign); } bufOutStream.close(); bufInStream.close(); } }
|
字符缓冲流
文件的复制(字符缓冲流,加速字符读取)
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24
| import java.io.*;
public class BufferedReaderWriterTest {
public static void main(String[] args) throws Exception { File readerFile = new File("D:\\test.txt") ; File writerFile = new File("D:\\test1.txt") ; BufferedReader bufReader = new BufferedReader(new FileReader(readerFile)) ; BufferedWriter bufWriter = new BufferedWriter(new FileWriter(writerFile)) ; String line; while ((line = bufReader.readLine()) != null){ bufWriter.write(line); bufWriter.newLine(); } bufWriter.flush(); bufWriter.close(); bufReader.close(); } }
|
转换流
转换流可以将字节流读取到的字节,按指定字符集解码成字符。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29
| import java.io.*; import java.nio.charset.StandardCharsets;
public class InputOutputStreamReaderTest {
public static void main(String[] args) { try { InputStreamReader isr = new InputStreamReader(new FileInputStream("D:\\test.txt")); BufferedReader br = new BufferedReader(isr);
FileOutputStream fos = new FileOutputStream("D:\\test.txt"); OutputStreamWriter osr = new OutputStreamWriter(fos, StandardCharsets.UTF_8); BufferedWriter bw = new BufferedWriter(osr);
int len; char[] c = new char[8 * 1024]; while ((len = br.read(c)) != -1) { bw.write(c, 0, len); bw.flush(); } } catch (Exception e) { e.printStackTrace(); } } }
|
数据流
数据流:用来处理基本数据类型、String、字节数组的数据。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19
| import java.io.*;
public class DataInputOutputStream {
public static void main(String[] args) throws Exception { DataOutputStream dataOutputStream = new DataOutputStream(new FileOutputStream("D:\\test.txt")); dataOutputStream.writeUTF("我爱你,而你不知道"); dataOutputStream.writeBoolean(true); dataOutputStream.writeInt(12); dataOutputStream.close();
DataInputStream dataInputStream = new DataInputStream(new FileInputStream("D:\\test.txt")); System.out.println(dataInputStream.readUTF()); System.out.println(dataInputStream.readBoolean()); System.out.println(dataInputStream.readInt()); dataInputStream.close(); } }
|
输出:
我爱你,而你不知道
true
12
对象流
序列化:对象转换为流的过程,反序列化:流转换为对象的过程。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32
| import java.io.*;
public class ObjectInputOutputStreamTest {
public static void main(String[] args) throws Exception { OutputStream outStream = new FileOutputStream("D:\\test.txt"); ObjectOutputStream objOutStream = new ObjectOutputStream(outStream); objOutStream.writeObject(new User(1,"test")); objOutStream.close();
InputStream inStream = new FileInputStream("D:\\test.txt"); ObjectInputStream objInStream = new ObjectInputStream(inStream) ; User user = (User) objInStream.readObject(); System.out.println(user); System.out.println(user.id); System.out.println(user.name); inStream.close(); }
static class User implements Serializable { private final Integer id ; private final String name ;
public User(int id, String name) { this.id = id; this.name = name; } } }
|
输出:
ObjectInputOutputStreamTest$User@47f6473
1
test
参考