java 文件操作

     在我的前面几篇文章中,我们介绍了流的概念,使用流我们可以任意读取写入文件中的内容,而对于文件的操作来说,更多的是对文件的存储进行操作,创建文件再磁盘上,移动文件到指定位置上,更改文件的文件名等。这些操作更多的是和操作系统以及文件系统打交道。首先我们看File类。      一个File类对象可以存放的是目录,也可以是文件。实际上在java 7开始,引进了Files类,集合Path接口实现了对整个文件操作的分离,Files类操作文件,Path操作路径。这篇文章先介绍File类。

一、File的构造方法

public File(String pathname)
public File(String parent, String child)
public File(File parent, String child)

     主要有三个构造方法,第一个构造方法允许传入一个表示路径的字符串,可以是绝对路径也可以是相对路径。内部调用文件系统类的方法为File对象中的实例域初始化。第二个构造方法允许传入两个字符串,由命名可以看出必然是可以拼接的,当然如果parent为空的话,自然是以child为路径初始化参数,这就等于第一种方式了。第三种构造方法,传入了一个File类的对象作为parent,其实在内部还是将此parent.path的路径值拿出来,执行的是和第二种构造方法一样的操作。

二、文件名和文件路径操作      我们知道,File对象既可以存文件也可以存路径,那么对他们的相关操作有如下几种:

public String getName()
public String getPath()
public boolean isAbsolute()

public String getParent()
public File getParentFile()

public String getAbsolutePath()
public File getAbsoluteFile()

public String getCanonicalPath()
public File getCanonicalFile()

     主要有上述几种对文件名和文件路径的操作。getName很简单,就是获取该文件对象的文件名,内部是怎么实现的呢?我们看看代码:

public String getName() {
        int index = path.lastIndexOf(separatorChar);
        if (index < prefixLength) return path.substring(prefixLength);
        return path.substring(index + 1);
    }

     separatorChar 表示路径分隔符,在windows中,一般默认使用“”,作为文件分隔符,Linux系统中使用“/”。prefixLength表示文件前缀名长度。(也就是最后一个路径分隔符前面的所有字符串的长度),此处index拿到最后一个路径分隔符的索引,截取此位置后面的字符串作为结果返回。这样无论File对象中存放的是什么,都可以拿到它的name。

public class Test_InputOrOutput {
    public static void main(String[] args) throws IOException{

        File f = new File("C://Users/directory");
        System.out.println(f.getName());

    }
}
//即使File对象中存放的是路径,依然返回路径名directory

     接下来看看getPath这个方法。getPath就一条语句,返回该对象的私有成员path,这是当初调用构造方法传入的pathName。isAbsolute方法调用文件系统类的方法判断当前pathName是否是绝对路径,getAbsolutePath获取当前File对象的绝对路径,相对于当前项目根目录的位置来指定的。      getParent方法返回当前文件对象的父目录路径。内部是这样实现的:

    public String getParent() {
        int index = path.lastIndexOf(separatorChar);
        if (index < prefixLength) {
            if ((prefixLength > 0) && (path.length() > prefixLength))
                return path.substring(0, prefixLength);
            return null;
        }
        return path.substring(0, index);
    }

     和getName方法类似,都是首先找到默认路径分隔符的最后一次出现的位置,然后通过一系列判断处理了各种意外的情况,例如:

File f = new File("directory");
System.out.println(f.getParent());  //输出:null

File f = new File("directory/");
System.out.println(f.getParent());  //输出:null

File f = new File("a/directory");
System.out.println(f.getParent());  //输出:a

     只有路径正常的时候才会执行:return path.substring(0, prefixLength);,返回对应的父目录的路径。getParentFile方法返回父目录对应的File对象。

public File getParentFile() {
        String p = this.getParent();
        if (p == null) return null;
        return new File(p, this.prefixLength);
    }

     从getParentFile的源代码中,我们可以看到,它调用了getParent方法获取父目录路径,但是如果该路径是空的的话就返回null,否则根据此路径构建一个新的File对象并返回。像下面这种情况就不能正确获取getParentFile:

File f = new File("directory");
File parent = f.getParentFile();
System.out.println(parent.getPath());
//抛出空指针异常,说明parent为null

     对于这种情况难道就没有办法了吗?其实可以组合getAbsoluteFile方法和getParentFile方法,来获取父目录的File对象。

File f = new File("directory");
File parent = f.getAbsoluteFile().getParentFile();
System.out.println(parent.getPath());

三、文件的信息      在我们的文件操作中,File对象中既可以存放文件有可以存放路径信息,那我们怎么区分他们呢?其实在File中还封装了一些判断文件信息的方法。

public boolean canRead()
public boolean canWrite()
public boolean exists()
public boolean isDirectory()
public boolean isFile()
public boolean isHidden()
public long lastModified()  //最后一次修改的时间

     从方法的命名大家就可以看出来每个方法所代表的含义。(世界上最好的注释就是没有注释,单命名就已经让人理解其作用)

四、操作文件      最后是文件的操作,真正意义上的对文件在磁盘上的存储方式进行操作。java中的File对象被创建出来之后,并不意味着在磁盘上已经创建了对应的文件,真正想要在磁盘上创建文件需要调用createNewFile方法。不用看实现代码,这么底层的操作肯定调用的是文件系统类中的方法。      使用createNewFile创建文件成功返回true,失败返回false。如果文件已经存在则不会去创建他,但是返回false。      文件的删除主要有两个方法。delete和deleteOnExit,前者会删除文件或者目录返回boolean,后者标记一下,等到虚拟机退出时候进行实际删除操作。需要注意的是如果将要删除的文件目录不为空就不能完成删除操作。

File f = new File("a");
f.mkdir();

File d = new File("a/a.txt");
d.createNewFile();

System.out.println(f.delete());
//输出结果:false

     所以在我们将要删除一个目录的时候就需要清空其中的所有内容。

五、目录操作      最后说说目录操作,其实在我们上面的代码中也已经稍有涉及了。创建一个目录有两种方法:

public boolean mkdir()
public boolean mkdirs()

     这两个操作有一定的区别,比如我要在c盘根目录下创建一个名为a的文件夹,new File("C://a").mkdir();或者new File("C://a").mkdirs();。浪着的效果是一样的。但是如果我要在c盘的b文件夹中创建一个d目录,其中b文件夹不存在。mkdir方法就无法完成任务了,因为中间有个文件夹b没有被创建。而mkdirs会将b文件夹也一起创建了。这就是它俩的区别。      关于目录操作的最后一个就是目录的遍历。File中也提供了很多方法。

public String[] list()
public File[] listFiles()

public String[] list(FilenameFilter filter)
public File[] listFiles(FilenameFilter filter)
public File[] listFiles(FileFilter filter)

     调用list返回目录下的所有文件的name:

File f = new File("f:/360");
String[] list = f.list();
for (String s : list){
    System.out.println(s);
}
//这是在我的f盘下的360文件夹中的所有文件名
输出结果:
360defender
360sdSetup.exe
360zip

     毋庸置疑,listFiles返回的是这些文件名构成的File对象数组。我们重点看剩下的几个方法,他们都传入了一个参数,这个参数其实就是过滤器,用来对目录下的文件进行进一步的筛选。他们两个都是函数式接口,只有一个方法。该方法返回true表示遍历中的当前的目录或者文件可以作为结果返回到结果集中。

public interface FileFiter{
    boolean accept(File pathName);
}

public interface FilenameFiter{
    boolean accept(File dir,String name);
}

     针对上述介绍,演示一个实例,使用的还是我的f盘下的360文件夹。

public static void main(String[] args) throws IOException{

        File f = new File("f:/360");

        File[] s = f.listFiles(new FileFilter() {

            @Override
            public boolean accept(File pathname) {
                return pathname.isDirectory() ? true : false;
            }
        });


        for (File name : s){
            System.out.println(name.getName());
        }
    }

     上述代码过滤筛选出,360文件夹中所有目录的文件,输出他们的名字。

     本篇文章结束,主要介绍了File类的一些使用情况,实际上java 7 中引进了Files类和Path接口实现了分离File类的作用,下篇文章我们一起探讨,本文若有不当之处,希望指出!

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏决胜机器学习

《Redis设计与实现》读书笔记(十八) ——Redis客户端属性设计与原理

《Redis设计与实现》读书笔记(十八) ——Redis客户端属性设计与原理 (原创内容,转载请注明来源,谢谢) 一、概述 redis服务器是...

2694
来自专栏Java工程师日常干货

写一个迷你版的Tomcat前言Write MyTomcat

Tomcat,这只3脚猫,大学的时候就认识了,直到现在工作中,也常会和它打交道。这是一只神奇的猫,今天让我来抽象你,实现你!

572
来自专栏用户2442861的专栏

Java IO学习笔记(一):File类

http://www.cnblogs.com/lich/archive/2011/12/10/2283445.html

401
来自专栏黑白安全

DLL注入新姿势:反射式DLL注入研究

在分析koadic渗透利器时,发现它有一个注入模块,其DLL注入实现方式和一般的注入方式不一样。搜索了一下发现是由HarmanySecurity的Stephen...

763
来自专栏好好学java的技术栈

http简介看这篇就够了

协议,网络协议的简称,网络协议是通信计算机双方必须共同遵从的一组约定。如怎么样建立连接、怎么样互相识别等。只有遵守这个约定,计算机之间才能相互通信交流。它的三要...

952
来自专栏好好学java的技术栈

「文末赠书」http协议简介看这篇就够了

协议,网络协议的简称,网络协议是通信计算机双方必须共同遵从的一组约定。如怎么样建立连接、怎么样互相识别等。只有遵守这个约定,计算机之间才能相互通信交流。它的三要...

733
来自专栏网络

深入理解php底层:php生命周期

文章来自:博客 http://blog.csdn.net/hguisu/article/details/7377520 1、PHP的运行模式: PHP两种运行模...

2238
来自专栏JAVA同学会

Spring Data(一)概念和仓库的定义

Spring Data的主要任务是为数据访问提供一个相似的、一致的、基于Spring的编程模型,同时又保留着下面各个数据存储的特征。它使得使用数据访问技术非常的...

611
来自专栏烂笔头

Python检查xpath和csspath表达式是否合法

目录[-] 在做一个可视化配置爬虫项目时,需要配置爬虫的用户自己输入xpath和csspath路径以提取数据或做浏览器操作。考虑到用户的有时会输入错误的x...

3399
来自专栏FreeBuf

如何利用Ptrace拦截和模拟Linux系统调用

ptrace(2)这个系统调用一般都跟调试离不开关系,它不仅是类Unix系统中本地调试器监控实现的主要机制,而且它还是strace系统调用常用的实现方法。ptr...

1187

扫码关注云+社区