Java操作文件

import java.io.File;
import java.io.IOException;
import java.nio.file.*;
import java.nio.file.attribute.BasicFileAttributes;

public class Test {
    public static void main(String[] args){
        WalkFileTreeTest.test();
    }
}

/*
    2.5 操作文件

    2.5.1 Path

    Paths.get()方法:

        //一个一个部件的写
        Path absolute = Paths.get("/home","harry");             :绝对路径/home/harry
        Path relative = Paths.get("myprog","conf","user.prop"); :相对路径./mypoog/conf/user.prop

        //传入一个包含多个部件的字符串
        Path basePath = Paths.get("/opt/myprog/");

    path.resolve()方法:
        //接受一个Path对象
        Path basePath = Paths.get("base");
        Path workPath1 = basePath.resolve(new Path(work));

        //接受一个字符串
        Path workPath2 = basePath.resolve("work2");

    path.resolveSibling()方法:
        //从path路径的父路径产生temp路径(兄弟路径)
        path.resolveSibling("temp");


    path.relativize()方法:
        //相对的处理(有道翻译)
        p.relativize(r),我觉得可以简单的理解为,获得r相对于p的路径。

    path.normalize()方法:
        //移除所有冗余的.和..部件
        /home/cay/../fred/./myprog  =>  /home/fred/myporg

    path.toAbsolutePath()方法:
        //返回path的绝对路径。

    Path将路径断开的各种方法:
        Path p = Paths.get("/home","fred","myprog.prop");

        Path parent = p.getParent();    //得到/home/fred
        Path file = p.getFileName();    //得到最后文件的名称
        Path root = p.getRoot();        //得到根路劲

    如果需要和遗留系统的API交互,可以使用的是File类。Path接口有一个toFile方法,
    而File类有一个toPath方法。
 */

/*
    2.5.2 读写文件

        byte[] bytes = Files.readAllBytes(path);                //读取一个文件的全部内容
        String content = new String(bytes,charset);             //将文件当做字符串读入
        List<String> lines = Files.readAllLines(path,charset)   //希望将文件当做行序列读入
        Files.write(path,content.getBytes(charset));            //希望写出一个字符串到文件中
        Files.write(path,content.getBytes(charset)
            ,StandOpenOption.Append);                           //希望在指定的文件后追加内容
        Files,write(path,lines)                                 //将一个行的集合,写进文件里

        得到相关的流:
            InputStream in = Files.newInputStream(path);
            OutputStream out = Files.newOutputStream(path);

            Reader in = Files.newBufferedReader(path,charset);
            Writer out = Files.newBufferedWriter(path,charset);
 */

/*
    2.5.3 创建文件和目录

        //创建一个空文件
        Files.createFile(path)

        //除路径最后一个部件外,其他部件都必须存在
        Files.createDirectory(path);
        //创建路径中的中间目录
        Files.createDirectories(path);


    用于在给定位置或系统指定位置创建零食文件或目录的方法:
        Path newPath = Files.createTimeFile(dir,prefix,suffix);
        Path newPath = Files.createTimeFile(prefix,suffix);
        Path newPath = Files.createTimeFile(dir,prefix);
        Path newPath = Files.createTimeFile(prefix);
 */

/*
    2.5.4 复制、移动和删除文件

        Files.copy(fromPath,toPath);
        Files.copy(fromPath,toPath,
            StandCopyOption.REPLACE_EXISTING,  //替换已经存在的文件
            StandCopyOption.COPY_ATTRIBUTES);  //复制所有的文件属性

        Files.move(fromPath,toPath);
        Files.move(fromPath,toPath,
            StandCopyOption.ATOMIC_MOVE);     //将移动操作定义为原子性

        Files.copy(inputStream,toPath);       //将输入流复制到Path中
        Files.copy(from,outputStream);        //将path复制到输出流

        Files.delete(path);                             //文件删除,会抛出异常
        boolean deleted = Files.deleteIfExists(path);   //不会抛出异常
 */

/*
                             StandardOpenOption
    ------------------------------------------------------------------------------------
    |    READ                用于文件读取而打开                                           |
    |    WRITE               用于文件写入而打开                                           |
    |    APPEND              如果用于文件写入而打开,在在文件末尾追加                         |
    |    TRUNCATE_EXISTING   如果用于文件写入而打开,那么移除已有内容                         |
    |    CREATE_NEW          创建新文件,文件存在的情况下会创建失败                           |
    |    CREATE              自动在文件不存在的情况下创建新文件                              |
    |    DELETE_ON_CLOSE     当文件被关闭时,尽可能的删除这个文件                            |
    |    SPARSE              给文件系统一个提示,该文件时稀疏的                              |
    |    DSYN|SYN            要求对文件数据|数据和元数据的每次更新都必须同步写入到存储设备中     |
    ------------------------------------------------------------------------------------
        备注:StandardOpenOption 与 newBufferedWriter newInputStream newOutputStream
                write一起使用


                                 StandardCopyOption
    ------------------------------------------------------------------------------------
    |    ATOMIC_MOVE         原子性的移动文件                                             |
    |    COPY_ATTRIBUTES     复制文件的属性                                              |
    |    REPLACE_EXISTIONG   如果目标已存在,则替换它                                      |
    ------------------------------------------------------------------------------------
        备注:StandardCopyOption 与 copy、move一起使用


                                     LinkOption
    ------------------------------------------------------------------------------------
    |    NOFOLLOW_LINKS         不要跟踪符号连接                                        |                                     |
    ------------------------------------------------------------------------------------
        备注:LinkOption 与 exists isDeicretory isRegularFile一起使用

                                     FileVisitOption
    ------------------------------------------------------------------------------------
    |    FLLOW_LINKS            跟踪符号连接                                       |                                     |
    ------------------------------------------------------------------------------------
        备注:FileVisitOption 与 fink walk walkFileTree一起使用
 */

/*
    2.5.5 获取文件信息

    检查某个属性的结果:
        exists
        isHidden
        isReadable,isWritable,isExecutable
        isRegularFile,isDirectory,isSymbolicLink

    size方法将返回文件的字节数:
        long fileSize = Files.size(path);

    getOwner:
        将文件的拥有者作为java.nio.file.attribute.UserPrincipal的一个
        实例返回。

    所有的文件系统都会报告一个基本属性集,它们被封装在BasicFileAttribute接口中
    这部分信息与上述信息有部分重叠。基本文件属性包括:
        1.创建文件、最后一次访问以及最后一次修改文件的时间,这些时间都被
            表示为java.nio.file.attribute.FileTime
        2.文件是常规文件、目录还是符号连接,或者都不是
        3.文件尺寸
        4.文件主键,这是某种类的对象,具体所属类与文件系统相关,有可能是
            文件的唯一标识符,也可能不是。

        =>
            PosixFileAttribute attributes = Files.readAttributes(
                path, BasicFileAttribute.class);
 */

/*
    2.5.6 访问目录中的项

    try(DirectoryStream<Path> entries = Files.newDirectoryStream(dir)){
        for(Path entry : entries){
            //Process entries
        }
    }

    可以使用glob模式来过滤文件:
    try(DirectoryStream<Path> entries = Files.newDirectoryStream(dir,"*.java"))

    ============================================================================================
    = *     = 匹配路径组成部分的0个或多个字符  =    *.java匹配当前目录中的所有Java文件                  =
    = **    = 匹配跨目录边界的0个或多个字符   =     **.java匹配在所有目录中的Java文件                  =
    = ?     = 匹配一个字符                  =    ???.java 匹配所有三个字符的Java文件                 =
    = []    = 匹配一个字符集合               =    Test[0-9A-F]匹配Testx.java,其中x是一个16进制数    =
    = {}    = 匹配由逗号分割开的多个选项之一   =    *.{java,class}匹配所有的Java文件和class文件        =
    = \     = 转义字符                     =     *\**匹配所有文件名中包含*的文件                    =
    ============================================================================================

    如果向要访问某个目录的所有子孙成员,可以转而调用walkFileTree方法,并向其传递一个FileVistor类型
    的对象。这个对象会得到下列通知:
        1.遇到一个文件或目录时:
            FileVisitResult visitFile(T path, BasicFileAttribute)
        2.在一个目录被处理前:
            FileVisitResult preVisitDirectory(T dir, IOException ex)
        3.在一个目录被处理后:
            FileVisitResult postVisitDirectory(T dir, IOException ex)
        4.在试图访问文件或目录时发生错误,例如没有权限打开目录:
            FileVisitResult visitFileFailed(path, IOException ex)

    对于上述的每种情况,都可以指定是否希望执行下面的操作:
        1.继续访问下一个文件:
            FileVisitResult.CONTINUE
        2.继续访问,但是不再访问这个目录下的任何项了:
            FileVisitResult.SKIP_SUBTREE
        3.继续访问,但是不再访问这个文件的兄弟文件(和该文件同一目录下的文件)了:
            FileVisitResult.SKIP_SIBLINGS
        4.终止访问:
            FileVisitResult.TERMINATE
 */
class WalkFileTreeTest{
    public static void test(){
        try{
            Files.walkFileTree(Paths.get("."),new SimpleFileVisitor<Path>(){
                public FileVisitResult preVisitDirectory(Path path, BasicFileAttributes attrs)
                        throws IOException{
                    System.out.println(path);
                    return FileVisitResult.CONTINUE;
                }

                @Override
                public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException {
                    return FileVisitResult.CONTINUE;
                }

                @Override
                public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
                    return FileVisitResult.SKIP_SUBTREE;
                }
            });
        } catch (IOException e) {
            e.printStackTrace();
        }

    }

    //用于删除目录树
    public static void test2(){
        try{
            Files.walkFileTree(Paths.get("."),new SimpleFileVisitor<Path>(){
                @Override
                public FileVisitResult visitFile(Path file, BasicFileAttributes attrs) throws IOException {
                    Files.delete(file);
                    return FileVisitResult.CONTINUE;
                }

                @Override
                public FileVisitResult postVisitDirectory(Path dir, IOException exc) throws IOException {
                    if(exc!=null)
                        throw exc;

                    Files.delete(dir);
                    return FileVisitResult.CONTINUE;
                }
            });
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}

/*
    2.5.8 ZIP文件系统

    Paths类会在默认的文件系统中查找路径,也可以让其在别的文件系统中查找。

        FileSystem fs = FileSystems.newFileSystem(Paths.get(zipname),null);

        Files.copy(fs.getPath(sourceName),targetPath);

        FileSystem fs = FileSystems.newFileSystem(Paths.get(zipname),null)
        Files.walkFileTree(fs.getPath("/",new SimpleFileVsitor<Path>(){
            public FileVisitResult visitFile(Path file, BasicFileAttribute attrs)
                throws IOException {
                System.out.println(file);
                return FileVisitResult.CONTINUE;
            }});

 */

 

《Java核心技术卷二》笔记

上一篇:字节跳动不讲武德,居然笔试的时候出这种题目……


下一篇:SHELL:多文件的重命名和移动