我在使用 Java 7 的FileSystem
和Files
API 将二进制文件写入简单的 ZIP 存档时遇到问题。
执行写入操作时出现问题,该操作根本不会引发异常,文件未写入 ZIP 存档,但在运行时可用(Files.exists(backup)
返回 true,可以使用Files.readAllBytes(backup)
读取文件(。
当程序关闭并重新启动时,该文件不再可用。
片段
此方法应该创建任何路径的备份,无论谁是文件系统提供程序,仅在 ZIP 存档中的路径上"失败"。
/**
* Creates backup of path provided by 'file' parameter.
*
* @param file input file requiring backup creation
* @return backup file
* @throws java.io.IOException thrown in case of unsuccessful backup attempt
*/
public static Path createBackup(Path file) throws IOException {
FileSystem fileSystem = file.getFileSystem();
Path backup = fileSystem.getPath(file.toString() + ".BAK");
return Files.write(backup, Files.readAllBytes(file));
}
public static void main(String... args) {
try {
Path f = FileSystems.newFileSystem(Paths.get("a.zip"), null).getPath("file.bin");
Path backup = createBackup(f);
System.out.println(Files.exists(backup)); // prints "true"
System.out.println(new String(Files.readAllBytes(backup))); // prints its bytes
System.out.println(backup.toString()); // prints "file.bin.BAK"
} catch (IOException ex) {
System.err.println(ex);
}
}
但是该文件实际上并不存在于 ZIP 中。
编辑: 我已经设法让它工作,但有一个问题。下面的代码关闭文件系统,但写入正确。需要以某种方式"刷新"/"重新打开"文件系统。
public static Path createBackup(Path file) throws IOException {
try(FileSystem fileSystem = file.getFileSystem()) {
Path backup = fileSystem.getPath(file.toString() + ".BAK");
return Files.write(backup, Files.readAllBytes(file));
}
}
当保留原始方法并在完成所有操作后手动关闭文件系统时,它会删除 zip 文件并保留类似zipfstmp***.tmp
的内容并抛出:
java.nio.file.FileAlreadyExistsException: zipfstmp2666831581340533856.tmp -> a.zip
当 tmp 文件重命名为"a.zip"时,它是一个有效的修改存档。
您应该在创建文件系统的调用方中使用 try-with-resource 语句进行关闭。根本不需要在createBackup
方法中处理文件系统。
public static Path createBackup(Path file) throws IOException {
Path backup = file.resolveSibling(file.getFileName().toString()+".BAK");
return Files.copy(file, backup, StandardCopyOption.REPLACE_EXISTING);
}
public static void main(String... args) {
try(FileSystem fs = FileSystems.newFileSystem(Paths.get("a.zip"), null)) {
Path f = fs.getPath("file.bin");
Path backup = createBackup(f);
System.out.println(Files.exists(backup)); // prints "true"
System.out.println(new String(Files.readAllBytes(backup))); // prints its bytes
System.out.println(backup.toString()); // prints "file.bin.BAK"
} catch (IOException ex) {
System.err.println(ex);
}
}
下面是一个简单的示例,其中所有逻辑都在 main 方法中。
解决此问题的关键是使用 URI 显式声明要使用的文件系统提供程序类型。你可能想知道为什么罐子而不是拉链。在Java中,jar文件实际上是zip文件,但具有不同的文件扩展名。因此,java可以使用相同的机制来访问zip文件。
我尝试了二进制文件和文本文件,它似乎可以在我的 Win 10 机器上运行。只需更改 zip 文件的源目录 (///e:/a.zip( 以指向 zip 文件的位置。对我来说,该文件.bin.BAK被写入 eclipse 项目目录的根目录(您可能也想更改它以进行部署(。
import java.io.IOException;
import java.net.URI;
import java.nio.file.FileSystems;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.Paths;
import java.nio.file.StandardCopyOption;
import java.util.HashMap;
import java.util.Map;
public class Test {
public static void main(String... args) {
//Create URI to define the source as a Zip file.
//I've used jar because this is essentially a zip file.
Map<String, String> env = new HashMap<>();
env.put("create", "true");
URI uri = URI.create("jar:file:///e:/a.zip");
try{
//Build paths to source file in zip and externally.
Path pathInZipfile = FileSystems.newFileSystem(uri, env).getPath("/file.bin");
Path extPath = Paths.get("file.bin.BAK");
//If file does not exist create it.
if (! Files.exists(extPath))
Files.createFile(extPath);
//Copy the files over.
Files.copy( pathInZipfile, extPath,
StandardCopyOption.REPLACE_EXISTING );
}//end try
catch (IOException ex){
ex.printStackTrace();
}//enc catch
}//end main
}//end class