文件是操作系统对磁盘上数据的组织形式。文件包括文件路径和文件名,比如:

/Users/Calvin/Desktop/demo.txt
复制代码

文件名的后缀其实是文件名的一部分,文件不一定要有后缀,但是一定要有文件路径和文件名,后缀名只是为了让一些操作系统更好的分辨文件的类型,以便对文件进行正确的操作,真正进行操作时,应用一般会检验文件的 MIME 类型,验证它属于哪种文件,而不是简单地靠后缀名判断。

所有的文件,不管是什么后缀名,都是一堆在磁盘上的二进制数据。这些二进制数据需要被正确的解析,文件才能被正确的使用。比如 PPT 文件,我们也可以用文本编辑器打开它,但是文本编辑器并不能正确解析PPT,所以显示的是一堆乱码。

即使是压缩文件,其实也只是一个文件,它通过内部的组织,将很多文件的数据以及目录结构信息压缩到了一个文件中。

本文我们来学习一下 Java 中常用的文件和目录操作,我们会写一个 Java 程序,每讲解完一个操作,程序就多一个文件操作的功能,到最后我们就有一个包含了常用文件操作功能的程序了,后续遇到相关的开发任务可以直接拿来参考。

Java IO 中对文件的抽象

上面我们介绍了文件是操作系统组织磁盘上数据的形式,在 Java 传统的 BIO 中通过 File 类( java.io.File)对文件的抽象,让我们可以通过 File 类访问系统的文件系统。使用 File 类提供的方法,可以完成以下操作:

下面我们通过几个实际的例子来介绍下使用 File 类完成上面这些文件操作。有一点需要注意的是,通过 File 类只能访问文件和目录元数据。如果需要读取或写入文件内容,应该使用 Java IO 的 FileInputStream 和 FileOutputStream 来完成,这部分内容我们放到后面的章节再介绍。

使用 File 类完成文件和目录操作文件和环境变量分隔符

在使用 File 类操作文件和文件夹之前,我们先来看两个分隔符,一个叫文件路径分隔符,另外一个叫环境变量分隔符。之所以介绍这两种分隔符,原因是fileinputstream读取文件,它们在不同操作系统下的表示方式不一样。

比如说,文件路径分隔符在 Windows 系统里使用的是反斜线符合”” ,而在Unix 、Linux 的系统里使用的则是斜线符号 “”,比如同样一个文件路径在 Windows 和 Mac 系统下会分别表示为

// 在 Windows 里的路径
C:UsersAdminrDeskTopfile.txt
// 在 Linux 或者 Mac 里的路径
/Users/Admin/Desktop/file.txt
复制代码

所以为了提供跨平台的兼容性,针对文件路径的分隔符 Java 提供了静态变量 File.separator 表示文件路径分隔符,它会自动判断底层是什么操作系统返回正确的路径分隔符。

与文件路径分隔符有相似问题的还有环境变量的分隔符,也是在Windows 和 Linux 系统上有所不同,Windows 上使用分号”;”,Linux、Mac 这些系统上使用冒号”:”。

// Windows 上的环境变量
C:WindowsSystem32cmd.exe;D:Program FilesJavajdk1.8.0
// Linux 上的环境变量
/usr/local/bin:/usr/bin:/bin
复制代码

所以 Java 也提供了 File.pathSeparator 静态变量来帮我们处理环境变量分隔符在不同系统上的差异。

package com.example.learnfile;
import java.io.File;
public class SeparatorAppMain {
    public static void main(String[] args) {
        System.out.println("文件路径分隔符:" + File.separator);
        System.out.println("环境变量分隔符:" + File.pathSeparator);
    }
}
复制代码

上面这个例程我们可以运行试一下,在Mac 上会有如下输出,使用 Windows 的读者们可以自己运行试一下,看看会不会输出Windows 对应的文件路径和环境变量分隔符。

文件路径分隔符:/
环境变量分隔符::
复制代码

下面开始,我们通过编写一个简易的文件和目录操作类的形式讲解一下 File 提供的文件和目录操作功能。 首先我们给类添加两个属性

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Scanner;
import java.util.stream.Collectors;
public class CreateDirAndFileAppMain {
    public static final String ROOT = "." + File.separator;
    private static Scanner scanner = new Scanner(System.in);
 	...   
}
复制代码

创建 File 实例

Java 在对文件系统类进行任何操作之前,必须先创建一个 File 类的实例。

File dir = new File(ROOT + "file-demo/");
复制代码

File 类的构造方法以文件的路径作为参数,如果参数指定的是一个文件系统上不存在的文件或目录,构造方法也不会抛出异常而是会正常返回 File 类的实例,通过 File 实例我们再来判断文件/目录是否存在,进行文件操作等等。

创建文件夹

在了解所有文件和目录操作都是通过 File 实例的方法完成的这一点后,我们编写第一个工具方法,用 File 实例检查路径是否是目录,是且目录不存在则用 File 实例创建之。

    private static File createDir(String... restPaths) {
        String rest = joinRestDir(restPaths);
        System.out.println("将在" + ROOT + "下创建" + rest);
        File dir = new File(ROOT, rest);
        if (dir.exists() && dir.isDirectory()) {
            System.out.println("文件夹已经存在" + dir.toString());
            return dir;
        } else {
            boolean createSuccess = dir.mkdirs();
            if (createSuccess) {
                return dir;
            } else {
                throw new IllegalArgumentException("无法在" + ROOT + "下创建" + rest);
            }
        }
    }
    private static String joinRestDir(String... restPaths) {
        return Arrays.stream(restPaths).map(String::trim).collect(Collectors.joining(File.separator));
    }
复制代码

这个方法有以下几点需要注意

c文件流读取文件_fileinputstream读取文件_ifstream读取文件

CreateDir 方法的参数我们是从命令行用户的输入中读到的。

    private static File createDirs() {
        List pathList = new ArrayList();
        while (true) {
            System.out.println("请输入文件路径,如果为空则结束");
            String path = scanner.nextLine();
            if (path.isBlank()) {
                break;
            }
            pathList.add(path);
        }
        return createDir(pathList.toArray(new String[0]));
    }
复制代码

该方法最后把用户输入的逐层目录名放到数组里,传递给了我们上面的创建目录的工具方法。

重命名目录/文件

    private static File renameDir(File dir) {
        System.out.println("请输入新的文件夹的名字:");
        String newDirName = scanner.nextLine().trim();
        File newDir = new File(dir.getParentFile(), newDirName);
        boolean renameSuccess = dir.renameTo(newDir);
        if (renameSuccess) {
            System.out.println("改名为" + newDirName + "成功");
        } else {
            System.out.println("改名为" + newDirName + "失败");
            return null;
        }
        return newDir;
    }
复制代码

其实通过例程我们也能看出来,重命名也能完成移动文件夹的操作,只要我们指定一个不同的父级目录的 File 对象作为 renameTo 的参数即可。同样我们这里演示的是重命名文件夹,如果 File 实例是指向一个文件的,那自然可以完成文件的重命名和移动。

所以使用 File 实例的 renameTo 方法我们能够完成:

创建文件

如果 File 实例指向的文件在文件系统里不存在,那么使用其 createNewFile() 方法,就能在文件系统里创建(持久化)该文件。

File file = new File("/tmp/file-demo/file.txt");
file.createNewFile();
复制代码

这个我们不做过多解释,直接上我们创建文件的工具方法。

    private static String createFiles(File newDir) throws IOException {
        System.out.println("请输入文件名的前缀:");
        String fileName = scanner.next().trim();
        for (int i = 0; i < 20; i++) {
            File f = new File(newDir, fileName + i + ".txt");
            System.out.println("创建文件" + f.getName() + ": " + f.createNewFile());
        }
        return fileName;
    }
复制代码

删除文件

要删除文件,调用 File 的 delete() 方法。

File file = new File("/tmp/file-demo/file.txt");
result = file.delete()
复制代码

delete() 方法返回布尔值(true 或 false),表示删除是否成功。删除文件可能会因各种原因而失败,比如文件已打开、文件权限错误等。delete() 方法也能用于删除目录,但是只能删除不包含任何文件和子目录的空目录。 下面我们看一下删除文件的交互式演示程序。

    private static void deleteFiles(File newDir, String fileNameNew) {
        System.out.println("删除文件?");
		// 命令行里要输入 true 或者 false 只是是否删除文件
        boolean deleteFiles = scanner.nextBoolean();
        if (deleteFiles) {
            for (int i = 0; i < 20; i++) {
                File fn = new File(newDir, fileNameNew + i + ".txt");
                System.out.println("删除文件:" + fn.delete());
            }
        }
    }
复制代码

上面的演示程序,通过读取命令行里输入的 true ,来确认用户想删除后再完成删除操作。

用递归完成目录删除

File 实例的 delete() 方法只能在目录为空时删除目录,如果要删除包含文件和子目录的目录,我们必须先遍历目录并删除所有文件和子目录,然后才能删除根目录。

这个迭代必须递归执行,才能完成目录的删除。

public static boolean deleteDir(File dir){
    File[] files = dir.listFiles();
    if(files != null){
        for(File file : files){
            if(file.isDirectory()){
                deleteDir(file);
            } else {
                file.delete();
            }
        }
    }
    return dir.delete();
}
复制代码

上面这个方法使用了一个还未介绍的 listFiles(),不过相信你已经猜到它的作用了。File 实例的 listFiles() 方法能列出File实例指向的目录下的所有文件和子目录,然后我们遍历删除目录下的文件,如果遇到子目录则再次调用 deleteDir 方法完成子目录内文件的删除,最终,通过这种递归的方式完成整个目录的删除。

同样我们也写个交互确认删除文件夹的函数达到演示效果。

    private static void removeDir(File dir) {
        System.out.println("删除文件夹?");
        boolean deleteDir = scanner.nextBoolean();
        if (deleteDir) {
            deleteDir(dir);
        }
    }
复制代码

执行演示程序

介绍完 File 类的方法操作文件和目录的功能后,我们整个演示程序就写完了fileinputstream读取文件,最后补充执行它的 Main 方法,让我们能运行这个演示程序

import java.io.File;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.Scanner;
import java.util.stream.Collectors;
public class CreateDirAndFileAppMain {
    // TODO 不同操作系统可以更改这个值,比如mac或者linux可以写为~代表home目录
    public static final String ROOT = "." + File.separator;
    private static Scanner scanner = new Scanner(System.in);
    public static void main(String[] args) throws IOException{
        // TODO 使用File类,依次创建多层文件夹,修改文件夹名字,在指定文件夹创建文件,删除文件,删除文件夹
        File dir = createDirs();
        File newDir = renameDir(dir);
        String fileName = createFiles(newDir);
        String fileNameNew = renameFiles(newDir, fileName);
        deleteFiles(newDir, fileNameNew);
        removeDir(newDir);
    }
    
    ......
}
复制代码

最后

本文整体学习下来,我们一起完成的完整可运行程序,因为篇幅太长,放在了下面的链接里,需要的可自行取用

限时特惠:本站每日持续更新海量设计资源,一年会员只需29.9元,全站资源免费下载
站长微信:ziyuanshu688