欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页

[NIO.2] 第二十九篇 删除、复制、移动目录和文件

程序员文章站 2022-04-25 08:24:52
...
删除、拷贝和移动操作是最常见的文件操作。NIO.2 提供了独立的方法来支持这些操作。它们中的大部分都来自 Files 类。

删除文件和目录

NIO.2 提供了两个方法来删除文件和目录,分别是  Files.delete() 和 Files.deleteIfExits()。这两个方法都接受一个 Path 类型的参数用于指定删除对象。不同的是,Files.delete() 的返回值是 void,而 Files.deleteIfExits() 的返回值是 boolean。

Files.delete() 方法试图删除传入的 Path 对象,如果删除失败,则有可能抛出下面的异常:NoSuchFileException (如果传入的 Path 文件/目录不存在),DirectoryNotEmptyException (如果传入的 Path 对象是一个目录,并且目录不为空),IOException (如果发生 I/O 异常), SecurityException (如果没有权限进行删除操作)。

下面的代码段将删除  C:\rafaelnadal\photos\ 目录下的 rafa_1.jpg 文件(文件必须存在)。

Path path = FileSystems.getDefault().getPath("C:/rafaelnadal/photos", "rafa_1.jpg"); 
 
//delete the file 
try { 
     Files.delete(path); 
} catch (NoSuchFileException | DirectoryNotEmptyException | IOException |  
         SecurityException e) { 
     System.err.println(e); 
}


如果使用 Files.deleteIfExists() 删除文件,文件必须存在,否则会返回 false(用于取代抛出 NoSuchFileException 异常)。这在多线程的环境下删除文件很有用,因为你可能会不希望线程中抛出异常。

下面的代码同样会删除 rafa_1.jpg 文件:

try { 
    boolean success = Files.deleteIfExists(path); 
    System.out.println("Delete status: " + success); 
} catch (DirectoryNotEmptyException | IOException | SecurityException e) { 
    System.err.println(e); 
}


注:如果要删除目录,那么目录必须为空,否则需要先进行递归删除目录下的所有内容。

复制文件和目录

复制是 NIO.2 提供的最美妙的方法之一。NIO.2 一共提供了三个 copy() 方法来完成这个任务,并且提供了一组选项来控制复制过程。copy() 方法是用非定长参数来接受这些选项。这些选项保存在 StandardCopyOption 和 LinkOption 枚举类型中,如下所示:

  •     REPLACE_EXISTING - 如果文件已经存在,则会被替换(如果复制的是非空目录,则会抛出  FileAlreadyExistsException 异常);如果复制的是软链接,则只是复制软链接本身,而不会复制目标文件。
  •     COPY_ATTRIBUTES - 赋值文件并且连同文件属性(至少会有 lastModifiedTime 属性支持复制)。
  •     NOFOLLOW_LINKS - 不考虑软链接目标文件,只考虑软链接自身。


如果你对枚举类型不熟,那么你可以使用下面的静态导入方式导入到你的类中:
import static java.nio.file.StandardCopyOption.REPLACE_EXISTING; 
import static java.nio.file.StandardCopyOption.COPY_ATTRIBUTES; 
import static java.nio.file.LinkOption.NOFOLLOW_LINKS;


注:默认情况下,如果被复制的文件是软连接,那么复制的将会是目标文件,只有使用了 REPLACE_EXISTING 和 NOFOLLOW_LINKS 选项才会赋值软链接本身。另外,文件属性默认不会被复制。

注意:试图复制一个非空目录最终会得到一个空目录。非空目录的复制需要进行递归复制下面的子目录和文件。另外,赋值文件的操作并不是一个原子操作,因此有可能会发生 IOException 或者赋值过程被打断的情况。

在两个 Path 对象之间复制

NIO.2 提供了 copy() 方法进行拷贝,它接受两个 Path 对象用于指定源文件和目标文件,还有一组用于控制复制过程的参数。返回目标文件 Path。默认情况下,若目标文件已存在,复制会失败。

下面的代码将会复制 draw_template.txt 文件(文件必须存在),从 C:\rafaelnadal\grandslam\AustralianOpen 复制到  C:\rafaelnadal\grandslam\USOpen。它将替换已有文件,并且会复制文件属性,不支持软链接。

Path copy_from = Paths.get("C:/rafaelnadal/grandslam/AustralianOpen", "draw_template.txt"); 
Path copy_to= Paths.get("C:/rafaelnadal/grandslam/USOpen",copy_from.getFileName().toString()); 
 
try { 
 
    Files.copy(copy_from, copy_to, REPLACE_EXISTING, COPY_ATTRIBUTES, NOFOLLOW_LINKS); 
 
} catch (IOException e) { 
    System.err.println(e); 
}


从输入流复制到文件

可以调用 Files.copy() 方法从输入流复制到文件,这个方法会返回读取或写出的字节数。默认情况下,如果目标文件已经存在,则复制失败。

下面代码段将使用输入流将文件  draw_template.txt 从 C:\rafaelnadal\grandslam\AustralianOpen 复制到 C:\rafaelnadal\grandslam\Wimbledon 并替换已有文件。

Path copy_from = Paths.get("C:/rafaelnadal/grandslam/AustralianOpen", "draw_template.txt"); 
Path copy_to = Paths.get("C:/rafaelnadal/grandslam/Wimbledon", "draw_template.txt"); 
 
try (InputStream is = new FileInputStream(copy_from.toFile())) { 
 
     Files.copy(is, copy_to, REPLACE_EXISTING); 
 
} catch (IOException e) { 
     System.err.println(e); 
}


输入流可以通过另外的途径获得,例如,下面的代码将会从 URL 获取输入流,并复制到 C:\rafaelnadal\photos 目录下(使用默认选项,文件必须不存在):

Path copy_to = Paths.get("C:/rafaelnadal/photos/rafa_winner_2.jpg"); 
URI u = URI.create("https://lh6.googleusercontent.com/-- 
                        udGIidomAM/Tl8KTbYd34I/AAAAAAAAAZw/j2nH24PaZyM/s800/rafa_winner.jpg"); 
 
try (InputStream in = u.toURL().openStream()) { 
 
     Files.copy(in, copy_to); 
 
} catch (IOException e) { 
     System.err.println(e); 
}


复制文件到输出流

可以调用 Files.copy() 方法从文件复制到输出流,这个方法会返回读取或写出的字节数。

下面的代码会将文件 draw_template.txt 从 C:\rafaelnadal\grandslam\AustralianOpen 复制到 C:\rafaelnadal\grandslam\RolandGarros,复制过程使用输出流(如果目标文件存在,则会被覆盖)。

Path copy_from = Paths.get("C:/rafaelnadal/grandslam/AustralianOpen", "draw_template.txt"); 
Path copy_to = Paths.get("C:/rafaelnadal/grandslam/RolandGarros", "draw_template.txt"); 
 
try (OutputStream os = new FileOutputStream(copy_to.toFile())) { 
 
     Files.copy(copy_from, os); 
} catch (IOException e) { 
     System.err.println(e); 
}


移动文件和目录

可以使用 Files.move() 方法来移动文件,这个方法接受两个 Path 类型对象,分别表示源文件和目标文件。并接受一组可选的参数,用于控制移动过程。这些参数存在在 StandardCopyOption 枚举类型中:

    REPLACE_EXISTING - 如果目标文件已经存在,则会被覆盖;如果操作的是软连接,那么软链接将会被替换并且指向的目标文件不变。
    ATOMIC_MOVE - 保证文件移动的原子性,这将保证任何任何监控文件目录的操作都将访问到完整的文件。

默认情况下,如果目标文件已经存在,移动将会失败。当移动软链接时,移动的是软链接本身而非目标文件。

下面的代码将会移动 rafa_2.jpg 文件,从 C:\rafaelnadal 到 C:\rafaelnadal\photos,如果文件存在,则会覆盖,因为使用了 REPLACE_EXISTING 选项:
Path movefrom = FileSystems.getDefault().getPath("C:/rafaelnadal/rafa_2.jpg"); 
Path moveto = FileSystems.getDefault().getPath("C:/rafaelnadal/photos/rafa_2.jpg"); 
       
try { 
    Files.move(movefrom, moveto, StandardCopyOption.REPLACE_EXISTING); 
} catch (IOException e) { 
    System.err.println(e); 
}


你可以使用 Path.resolve()  方法来避免文件名硬编码。下面的代码演示从源文件提取文件名并移动到目标目录下:

Path movefrom = FileSystems.getDefault().getPath("C:/rafaelnadal/rafa_2.jpg"); 
Path moveto_dir = FileSystems.getDefault().getPath("C:/rafaelnadal/photos"); 
 
try { 
    Files.move(movefrom, moveto_dir.resolve(movefrom.getFileName()),  
                                                         StandardCopyOption.REPLACE_EXISTING); 
} catch (IOException e) { 
    System.err.println(e); 
}


重命名文件

最后,演示一下如何使用 Files.move() 和 Path.resolveSibling() 方法对文件进行重命名。下面的代码将 rafa_2.jpg 重命名为  rafa_renamed_2.jpg:

Path movefrom = FileSystems.getDefault().getPath("C:/rafaelnadal/photos/rafa_2.jpg"); 
 
try { 
    Files.move(movefrom, movefrom.resolveSibling("rafa_2_renamed.jpg"),  
                                                         StandardCopyOption.REPLACE_EXISTING); 
} catch (IOException e) { 
    System.err.println(e); 
}


文章来源:http://www.aptusource.org/2014/04/nio-2-delete-copy-move-files/
相关标签: Java NIO.2