Java 使用 File 类来直接处理文件和文件系统。File 类没有指定信息怎样从文件读取或向文件存储;它描述了文件本身的属性。File 对象用来获取或处理与磁盘文件相关的信息,例如权限,时间,日期和目录路径。此外,File 还浏览子目录层次结构。Java 中的目录当成 File 对待,它具有附加的属性——一个可以被 list()
方法检测的文件名列表
File 类提供了以下构造方法:
//根据 parent 抽象路径名和 child 路径名字符串创建一个新 File 实例。 File(File parent, String child) //通过将给定路径名字符串转换为抽象路径名来创建一个新 File 实例 File(String pathname) // 根据 parent 路径名字符串和 child 路径名字符串创建一个新 File 实例 File(String parent, String child) //通过将给定的 file: URI 转换为一个抽象路径名来创建一个新的 File 实例 File(URI uri)
使用用例:
//一个目录路径参数 File f1 = new File("/home/project/"); //对象有两个参数——路径和文件名 File f2 = new File("/home/project/","a.bat"); //指向f1文件的路径及文件名 File f3 = new File(f1,"a.bat");
来看看 File 的一些常用方法
方法 | 说明 |
---|---|
boolean canExecute() | 测试应用程序是否可以执行此抽象路径名表示的文件 |
boolean canRead() | 测试应用程序是否可以读取此抽象路径名表示的文件 |
boolean canWrite() | 测试应用程序是否可以修改此抽象路径名表示的文件 |
int compareTo(File pathname) | 按字母顺序比较两个抽象路径名 |
boolean createNewFile() | 当且仅当不存在具有此抽象路径名指定名称的文件时,不可分地创建一个新的空文件 |
static File createTempFile(String prefix, String suffix) | 在默认临时文件目录中创建一个空文件,使用给定前缀和后缀生成其名称 |
static File createTempFile(String prefix, String suffix, File directory) | 在指定目录中创建一个新的空文件,使用给定的前缀和后缀字符串生成其名称 |
boolean delete() | 删除此抽象路径名表示的文件或目录 |
void deleteOnExit() | 在虚拟机终止时,请求删除此抽象路径名表示的文件或目录 |
boolean equals(Object obj) | 测试此抽象路径名与给定对象是否相等 |
boolean exists() | 测试此抽象路径名表示的文件或目录是否存在 |
File getAbsoluteFile() | 返回此抽象路径名的绝对路径名形式 |
String getAbsolutePath() | 返回此抽象路径名的绝对路径名字符串 |
File getCanonicalFile() | 返回此抽象路径名的规范形式 |
String getCanonicalPath() | 返回此抽象路径名的规范路径名字符串 |
long getFreeSpace() | 返回此抽象路径名指定的分区中未分配的字节数 |
String getName() | 返回由此抽象路径名表示的文件或目录的名称 |
String getParent() | 返回此抽象路径名父目录的路径名字符串;如果此路径名没有指定父目录,则返回 null |
File getParentFile() | 返回此抽象路径名父目录的抽象路径名;如果此路径名没有指定父目录,则返回 null |
String getPath() | 将此抽象路径名转换为一个路径名字符串 |
long getTotalSpace() | 返回此抽象路径名指定的分区大小 |
long getUsableSpace() | 返回此抽象路径名指定的分区上可用于此虚拟机的字节数 |
int hashCode() | 计算此抽象路径名的哈希码 |
boolean isAbsolute() | 测试此抽象路径名是否为绝对路径名 |
boolean isDirectory() | 测试此抽象路径名表示的文件是否是一个目录 |
boolean isFile() | 测试此抽象路径名表示的文件是否是一个标准文件 |
boolean isHidden() | 测试此抽象路径名指定的文件是否是一个隐藏文件 |
long lastModified() | 返回此抽象路径名表示的文件最后一次被修改的时间 |
long length() | 返回由此抽象路径名表示的文件的长度 |
String[] list() | 返回一个字符串数组,这些字符串指定此抽象路径名表示的目录中的文件和目录 |
String[] list(FilenameFilter filter) | 返回一个字符串数组,这些字符串指定此抽象路径名表示的目录中满足指定过滤器的文件和目录 |
File[] listFiles() | 返回一个抽象路径名数组,这些路径名表示此抽象路径名表示的目录中的文件 |
File[] listFiles(FileFilter filter) | 返回抽象路径名数组,这些路径名表示此抽象路径名表示的目录中满足指定过滤器的文件和目录 |
File[] listFiles(FilenameFilter filter) | 返回抽象路径名数组,这些路径名表示此抽象路径名表示的目录中满足指定过滤器的文件和目录 |
static File[] listRoots() | 列出可用的文件系统根 |
boolean mkdir() | 创建此抽象路径名指定的目录 |
boolean mkdirs() | 创建此抽象路径名指定的目录,包括所有必需但不存在的父目录 |
boolean renameTo(File dest) | 重新命名此抽象路径名表示的文件 |
boolean setExecutable(boolean executable) | 设置此抽象路径名所有者执行权限的一个便捷方法 |
boolean setExecutable(boolean executable, boolean ownerOnly) | 设置此抽象路径名的所有者或所有用户的执行权限 |
boolean setLastModified(long time) | 设置此抽象路径名指定的文件或目录的最后一次修改时间 |
boolean setReadable(boolean readable) | 设置此抽象路径名所有者读权限的一个便捷方法 |
boolean setReadable(boolean readable, boolean ownerOnly) | 设置此抽象路径名的所有者或所有用户的读权限 |
boolean setReadOnly() | 标记此抽象路径名指定的文件或目录,从而只能对其进行读操作 |
boolean setWritable(boolean writable) | 设置此抽象路径名所有者写权限的一个便捷方法 |
boolean setWritable(boolean writable, boolean ownerOnly) | 设置此抽象路径名的所有者或所有用户的写权限 |
String toString() | 返回此抽象路径名的路径名字符串 |
URI toURI() | 构造一个表示此抽象路径名的 file: URI |
java.io
包中提供了文件操作类:
接下来将学习文件流的 FileInputStream 和 FileOutputStream 。
FileInputStream 类用于打开一个输入文件,若要打开的文件不存在,则会产生异常 FileNotFoundException,这是一个非运行时异常,必须捕获或声明抛弃;
在进行文件的读 / 写操作时,会产生非运行时异常 IOException,必须捕获或声明抛弃(其他的输入 / 输出流处理时也同样需要进行输入 / 输出异常处理)。
//打开一个以 f 描述的文件作为输入 FileInputStream(File f) //打开一个文件路径名为 name 的文件作为输入 FileInputStream(String name) //创建一个以 f 描述的文件作为输出 //如果文件存在,则其内容被清空 FileOutputStream(File f) //创建一个文件路径名为 name 的文件作为输出 //文件如果已经存在,则其内容被清空 FileOutputStream(String name) //创建一个文件路径名为 name 的文件作为输出 //文件如果已经存在,则在该输出上输出的内容被接到原有内容之后 FileOutputStream(String name, boolean append)
输入流的参数是用于指定输入的文件名,输出流的参数则是用于指定输出的文件名。
如果文件内容保存的是字符信息,如 txt
文件等,还可以使用 FileReader 来读取文件内容。
FileReader file = new FileReader("/home/project/shiyanlou.txt"); //声明一个文件输入流file,并指明该文件在系统中的路径以方便定位 int data = 0; //声明一个整型变量用于存放读取的数据 while((data=file.read())!=-1){ //在while循环中使用read()方法持续读取file,数据赋到data中 //如果读取失败或者结束,则将返回-1,这个特殊的返回值可以作为读取结束的标识 System.out.print((char)data); //输出读取到数据 } file.close(); //一定要记得读取结束后要关闭文件
对于 FileInputStream/FileOutputStream、FileReader/FileWriter 来说,它们的实例都是顺序访问流,即只能进行顺序读 / 写。而类 RandomAccessFile 则允许文件内容同时完成读和写操作,它直接继承 Object,并且同时实现了接口 DataInput 和 DataOutput。
随机访问文件的行为类似存储在文件系统中的一个大型 byte
数组。存在指向该隐含数组的光标或索引,称为文件指针;输入操作从文件指针开始读取字节,并随着对字节的读取而前移此文件指针。如果随机访问文件以读取 / 写入模式创建,则输出操作也可用;输出操作从文件指针开始写入字节,并随着对字节的写入而前移此文件指针。
RandomAccessFile 提供了支持随机文件操作的方法:
readXXX()
或者 writeXXX()
: 如 readInt()
, readLine()
, writeChar()
, writeDouble()
等。int skipBytes(int n)
: 将指针向下移动若干字节。int length()
: 返回文件长度。long getFilePointer()
: 返回指针当前位置。void seek(long pos)
: 将指针调用所需位置。在生成一个随机文件对象时,除了要指明文件对象和文件名之外,还需要指明访问文件的模式。
来看看 RandomAccessFile 的构造方法:
RandomAccessFile(File file,String mode)
RandomAccessFile(String name,String mode)
mode 的取值:
r
: 只读,任何写操作都讲抛出 IOExceptionrw
: 读写,文件不存在时会创建该文件,文件存在是,原文件内容不变,通过写操作改变文件内容。rws
: 打开以便读取和写入,对于 "rw"
,还要求对文件的内容或元数据的每个更新都同步写入到底层存储设备。rwd
: 打开以便读取和写入,对于 "rw"
,还要求对文件内容的每个更新都同步写入到底层存储设备。可以使用 Files 工具类的 copy(Path source,Path target,CopyOption... options)
拷贝文件或者目录。如果目标文件存在,那么赋值将失败,除非我们在 options
中指定了 REPLACE_EXISTING
属性。当该命令复制目录时,如果目录中已经有了文件,目录中的文件将不会被复制。CopyOption 参数支持以下 StandardCopyOption 和 LinkOption 枚举:
REPLACE_EXISTING
:即使目标文件已存在,也执行复制。如果目标是符号链接,则复制链接本身(而不是链接的目标)。如果目标是非空目录,则复制将失败并显示 FileAlreadyExistsException 异常。COPY_ATTRIBUTES
:将与文件关联的文件属性复制到目标文件。支持的确切 - 文件属性是文件系统和平台相关的,但 last-modified-time 跨平台支持并复制到目标文件。 NOFOLLOW_LINKS
:表示不应遵循符号链接。如果要复制的文件是符号链接,则复制链接(而不是链接的目标)。
Files 类的 move(Path, Path, CopyOption... options)
方法移动文件或者目录,同样目标目录存在,那么比如使用REPLACE_EXISTING
。 options
参数支持 StandardCopyOption 的以下枚举:
REPLACE_EXISTING
:即使目标文件已存在,也执行移动。如果目标是符号链接,则替换符号链接,但它指向的内容不受影响。ATOMIC_MOVE
:将移动作为原子文件操作执行。如果文件系统不支持原子移动,则抛出异常。使用,ATOMIC_MOVE
您可以将文件移动到目录中,并保证观察目录的任何进程都可以访问完整的文件。move 方法除了可以移动之外,也可以用与重命名。
可以通过 Files 的 delete(Path path)
方法或者 deleteIfExists(Path path)
方法删除文件。
Java 使用 File 类表示文件或者目录,可以通过 File 类获取文件或者目录的相关属性
Java 中读取目录中的文件可以直接使用 listFiles()
方法读取,但是也只能读取当前目录中的文件,如果当前目录中还有二级目录如何解决呢?三级目录呢?接下来将使用 Java 读取当前目录和子目录中的所有文件
在大多数程序中,都需要对输入输出进行处理。例如我们中需要获取用户从键盘上的输入,需要在控制台输出结果等等。除此之外还有从文件中读取数据,向文件中写入数据等等。在 Java 中,我们把这些不同类型的输入输出源抽象地称为 _流_,也就是 Stream
;在里面输入输出的数据则称为数据流(Data Stream
),它们通常具有统一的接口。
于是我们得到了数据流的定义:
一个 Java I/O 对象叫做数据流。读取数据到内存的对象叫做输入流,内存写出数据的对象叫做输出流。
针对其面向的不同角度,我们大致可以将流分为下面几种类型:
需要特别说明,节点流是从特定的数据节点(文件、数据库、内存等)读写数据;处理流是连接在已有的流上,通过对数据的处理为程序提供更多功能。
在 Java 环境中,java.io
包提供了大多数的类和接口来实现输入输出管理。一些标准的输入输出则来自 java.lang
包中的类,但它们都是继承自 java.io
中的类。我们可以将输入流理解为数据的提供者,而把输出流理解为数据的接收者。在最初的时候,这些派生自抽象类 InputStream
和 OutputStream
的输入输出类是面向 8 位的字节流的。但为了支持国际化,又引入了派生自抽象类 Reader
和 Writer
的类层次,用于读写一些双字节的 Unicode
字符。
因此,在学习 java 的输入输出上,我们希望你以字节流和字符流作为区分来学习。
如果需要概括一下,则可以得到下面的定义:
按照这样的定义,Java 中流的层级结构可以通过下图来表示:
图中蓝色的部分均为抽象类,而绿色的部分则为派生类,是可以直接使用的。
而下图简要说明了字节流和字符流的区别,你也可以进一步了解字节流与字符流的区别。
我们知道 Java 是一门面向对象的语言,所以为了能够永久地保存对象的状态,java.io
包还以字节流为基础,通过实现 ObjectInput
和 ObjectOutput
接口提供了 _对象流_。在此仅作引入,你可以通过查阅 API 手册来详细了解它们。
字节流主要操作 byte 类型数据,以 byte
数组为准,java 中每一种字节流的基本功能依赖于基本类 InputStream 和 Outputstream,他们是抽象类,不能直接使用。字节流能处理所有类型的数据(如图片、avi 等)。
InputStream 是所有表示字节输入流的父类,继承它的子类要重新定义其中所定义的抽象方法。InputStream 是从装置来源地读取数据的抽象表示,例如 System 中的标准输入流 in
对象就是一个 InputStream 类型的实例。
InputStream 类方法:
方法 | 说明 |
---|---|
read()throws IOException | 从输入流中读取数据的下一个字节(抽象方法) |
skip(long n) throws IOException | 跳过和丢弃此输入流中数据的 n 个字节 |
available()throws IOException | 返回流中可用字节数 |
mark(int readlimit)throws IOException | 在此输入流中标记当前的位置 |
reset()throws IOException | 将此流重新定位到最后一次对此输入流调用 mark 方法时的位置 |
markSupport()throws IOException | 测试此输入流是否支持 mark 和 reset 方法 |
close()throws IOException | 关闭流 |
在 InputStream 类中,方法 read()
提供了三种从流中读数据的方法:
int read()
:从输入流中读一个字节,形成一个 0~255 之间的整数返回(是一个抽象方法)int read(byte b[])
:从输入流中读取一定数量的字节,并将其存储在缓冲区数组 b
中。int read(byte b[],int off,int len)
:从输入流中读取长度为 len
的数据,写入数组 b
中从索引 off
开始的位置,并返回读取得字节数。对于这三个方法,若返回 -1,表明流结束,否则,返回实际读取的字符数。
OutputStream 是所有表示位输出流的类之父类。子类要重新定义其中所定义的抽象方法,OutputStream 是用于将数据写入目的地的抽象表示。例如 System 中的标准输出流对象 out 其类型是 java.io.PrintStream
,这个类是 OutputStream 的子类。
OutputStream 类方法:
方法 | 说明 |
---|---|
write(int b)throws IOException | 将指定的字节写入此输出流(抽象方法) |
write(byte b[])throws IOException | 将字节数组中的数据输出到流中 |
write(byte b[], int off, int len)throws IOException | 将指定 byte 数组中从偏移量 off 开始的 len 个字节写入此输出流 |
flush()throws IOException | 刷新此输出流并强制写出所有缓冲的输出字节 |
close()throws IOException | 关闭流 |
字符流以字符为单位,根据码表映射字符,一次可能读多个字节,只能处理字符类型的数据。
java.io 包中专门用于字符流处理的类,是以 Reader 和 Writer 为基础派生的一系列类。
同类 InputStream 和 OutputStream 一样,Reader 和 Writer 也是抽象类,只提供了一系列用于字符流处理的接口。它们的方法与类 InputStream 和 OutputStream 类似,只不过其中的参数换成字符或字符数组。
Reader 是所有的输入字符流的父类,它是一个抽象类。
我们先来看一看基类 Reader 的方法,其用法与作用都与 InputStream 和 OutputStream 类似,就不做过多的说明了。
方法 | 返回值 |
---|---|
close() | void |
mark (int readAheadLimit) | void |
markSupported() | boolean |
read() | int |
read(char[] cbuf, int off,int len) | int |
ready() | boolean |
reset() | void |
skip(long n) | long |
Writer 是所有的输出字符流的父类,它是一个抽象类。Writer 的方法:
方法 | 返回值 |
---|---|
close() | void |
flush() | void |
write(char[] cbuf) | void |
write(char[] cbuf, int off,int len) | void |
write(int c) | void |
write(String str) | void |
write(String str, int off, int len) | void |
在这里我们就列举一下有哪些类。
类有千万,方法更是不计其数,所以没有必要去掌握所有的方法和类,只需要知道常见常用的就行了,而大多数的类和方法,希望大家有一个印象,当我们在实际开发的时间,能够想到,并且借助其他工具去查询我们需要的方法的应用方式就可以了。
InputStreamReader 和 OutputStreamWriter 是 java.io
包中用于处理字符流的最基本的类,用来在字节流和字符流之间作为中介:从字节输入流读入字节,并按编码规范转换为字符;往字节输出流写字符时先将字符按编码规范转换为字节。使用这两者进行字符处理时,在构造方法中应指定一定的平台规范,以便把以字节方式表示的流转换为特定平台上的字符表示。
InputStreamReader(InputStream in); //缺省规范说明 //指定规范 enc InputStreamReader(InputStream in, String enc); OutputStreamWriter(OutputStream out); //缺省规范说明 //指定规范 enc OutputStreamWriter(OutputStream out, String enc);
如果读取的字符流不是来自本地时(比如网上某处与本地编码方式不同的机器),那么在构造字符输入流时就不能简单地使用缺省编码规范,而应该指定一种统一的编码规范“ISO 8859_1”,这是一种映射到 ASCCII 码的编码方式,能够在不同平台之间正确转换字符。
InputStreamReader ir = new InputStreamReader(is,"8859_1");
buf
是一个位数组,默认为 2048 字节。当读取数据来源时例如文件,BufferedInputStream 会尽量将 buf
填满。当使用 read()
方法时,实际上是先读取 buf
中的数据,而不是直接对数据来源作读取。当 buf 中的数据不足时,BufferedInputStream 才会再实现给定的 InputStream 对象的 read()
方法,从指定的装置中提取数据。BufferedOutputStream 的数据成员 buf
是一个位数组,默认为 512 字节。当使用 write()
方法写入数据时,实际上会先将数据写至 buf
中,当 buf
已满时才会实现给定的 OutputStream 对象的 write()
方法,将 buf
数据写至目的地,而不是每次都对目的地作写入的动作
//[ ]里的内容代表可选参数 BufferedInputStream(InputStream in [, int size]) BufferedOutputStream(OutputStream out [, int size])
同样的,为了提高字符流处理的效率,java.io 中也提供了缓冲流 BufferedReader 和 BufferedWrite。其构造方法与 BufferedInputStream 和 BufferedOutPutStream 相类似。另外,除了 read() 和 write() 方法外,它还提供了整行字符处理方法:
\n
、\r
或者两者一起(这是根据系统而定的)\n
或\r
,而是系统定义的行隔离标志(line separator)。DataInput和DataOutput,设计了一种较为高级的数据输入输出方式:除了可处理字节和字节数组外,还可以处理
int、
float、
boolean等基本数据类型,这些数据在文件中的表示方式和它们在内存中的一样,无须转换,如
read(),
readInt(),
readByte()...;
write(),
writeChar(),
writeBoolean()... 此外,还可以用
readLine()
方法读取一行信息。方法 | 返回值 | 说明 |
---|---|---|
readBoolean() | boolean | |
readByte() | byte | |
readShort() | short | |
readChar() | char | |
readInt() | int | |
readLong() | long | |
readDouble() | double | |
readFloat() | float | |
readUnsignedByte() | int | |
readUnsignedShort() | int | |
readFully(byte[] b) | void | 从输入流中读取一些字节,并将它们存储在缓冲区数组 b 中 |
reaFully(byte[] b, int off,int len) | void | 从输入流中读取 len 个字节 |
skipBytes(int n) | int | 与 InputStream.skip 等价 |
readUTF() | String | 按照 UTF-8 形式从输入中读取字符串 |
readLine() | String | 按回车 (\r) 换行 (\n) 为分割符读取一行字符串,不完全支持 UNICODE |
writeBoolean(boolean v) | void | |
writeByte(int v) | void | |
writeShort(int v) | void | |
writeChar(int v) | void | |
writeInt(int v) | void | |
writeLong(long v) | void | |
writeFloat(float v) | void | |
writeDouble(double v) | void | |
write(byte[] b) | void | 与 OutputStream.write 同义 |
write(byte[] b, int off, int len) | void | 与 OutputStream.write 同义 |
write(int b) | void | 与 OutputStream.write 同义 |
writeBytes(String s) | void | 只输出每个字符的低 8 位;不完全支持 UNICODE |
writeChars(String s) | void | 每个字符在输出中都占两个字节 |
数据流类 DataInputStream
和 DataOutputStream
的处理对象除了是字节或字节数组外,还可以实现对文件的不同数据类型的读写:
DataInput
和 DataOutput
接口。boolean
,int
,long
,double
等基本数据类型,并可以再次把基本数据类型的值读取回来。数据流可以连接一个已经建立好的数据对象,例如网络连接、文件等。数据流可以通过如下方式
FileInputStream fis = new FileInputStream("file1.txt"); FileOutputStream fos = new FileOutputStream("file2.txt"); DataInputStream dis = new DataInputStream(fis); DataOutputStream dos = new DataOutputStream(fos);
Java NIO(New IO) 发布于 JDK1.4,用于代替 Java 标准 IO 。Java NIO 是面向缓存的、非阻塞的 IO,而标准 IO 是面向流的,阻塞的 IO。
首先理解 NIO 的重要概念:Buffer(缓冲区)
allocate()
方法分配 Buffer,Buffer 不可实例化,Buffer 是抽象类,需要使用具体的子类,比如 ByteBuffer。capacity
:缓冲区的容量position
:当前指针位置,每读取一次缓冲区数据或者写入缓冲区一个数据那么指针将会后移一位limit
:限制指针的移动,指针不能读取 limit
之后的位置mark
:如果设置该值,那么指针将移动到 0 - position
的位置mark
<= position
<= limit
<= capacity
原文:https://www.cnblogs.com/cheng-hong/p/14032250.html