lib/modules
中。这是我使用的基本Maven项目资源结构:
src
main
resources
dict
xkcd_en
我只是想读取xkcd_en
文本文件。如果我们查看JRT文件,它在这里:
>> jimage list /path/to/lib/modules
...
Module: main
dict/xkcd_en
...
此外,我已经在module-info
中明确打开了它,以防万一:
module main {
opens dict;
// ..rest code omitted
}
我只能将文件作为输入流来读取:
可行:
public static InputStream getResourceAsStream(String resource) {
return FileUtils.class.getResourceAsStream(resource);
}
System.out.println(new BufferedReader(
new InputStreamReader(getResourceAsStream("/dict/xkcd_en")))
.lines().collect(Collectors.joining("\n"))
);
无法正常工作:
但如果我尝试使用Java NIO API获取文件URI并读取它,它无法正常工作:
public static URL getResourceOrThrow(String resource) {
URL url = FileUtils.class.getResource(resource);
Objects.requireNonNull(url);
return url;
}
1 - Java NIO找不到文件。但是它绝对存在,否则getResource()
会返回null
。
System.out.println(Paths.get(getResourceOrThrow("/dict/xkcd_en").toURI()));
// /main/dict/xkcd_en
Files.readAllLines(Paths.get(getResourceOrThrow("/dict/xkcd_en").toURI()));
Caused by: java.nio.file.NoSuchFileException: /main/dict/xkcd_en
at java.base/jdk.internal.jrtfs.JrtFileSystem.checkNode(JrtFileSystem.java:494)
at java.base/jdk.internal.jrtfs.JrtFileSystem.getFileContent(JrtFileSystem.java:253)
at java.base/jdk.internal.jrtfs.JrtFileSystem.newInputStream(JrtFileSystem.java:342)
at java.base/jdk.internal.jrtfs.JrtPath.newInputStream(JrtPath.java:631)
at java.base/jdk.internal.jrtfs.JrtFileSystemProvider.newInputStream(JrtFileSystemProvider.java:322)
2 - 如果您直接使用FileSystem
,则情况相同:
FileSystem fs = FileSystems.getFileSystem(URI.create("jrt:/"));
System.out.println(fs.getPath("main/dict/xkcd_en"));
// main/dict/xkcd_en
Files.readAllLines(fs.getPath("main/dict/xkcd_en")));
Caused by: java.nio.file.NoSuchFileException: /main/dict/xkcd_en
at java.base/jdk.internal.jrtfs.JrtFileSystem.checkNode(JrtFileSystem.java:494)
3 - Java NIO甚至不知道jrt:/
协议是什么。
Files.readAllLines(Paths.get(getResourceOrThrow("/dict/xkcd_en").toExternalForm()));
Caused by: java.nio.file.InvalidPathException: Illegal char <:> at index 3: jrt:/main/dict/xkcd_en
at java.base/sun.nio.fs.WindowsPathParser.normalize(WindowsPathParser.java:182)
at java.base/sun.nio.fs.WindowsPathParser.parse(WindowsPathParser.java:153)
at java.base/sun.nio.fs.WindowsPathParser.parse(WindowsPathParser.java:77)
at java.base/sun.nio.fs.WindowsPath.parse(WindowsPath.java:92)
at java.base/sun.nio.fs.WindowsFileSystem.getPath(WindowsFileSystem.java:229)
at java.base/java.nio.file.Path.of(Path.java:147)
at java.base/java.nio.file.Paths.get(Paths.java:69)
这里是 JRT FS 的 规范。
jrt URL 是一个分层的 URI,符合 RFC 3986,其语法如下:
jrt:/[$MODULE[/$PATH]]
其中 $MODULE 是可选的模块名称,$PATH(如果存在)是该模块内特定类或资源文件的路径。jrt URL 的含义取决于其结构:
- jrt:/$MODULE/$PATH 引用给定 $MODULE 中名为 $PATH 的特定类或资源文件。
- jrt:/$MODULE 引用 $MODULE 模块中的所有类和资源文件。
- jrt:/ 引用存储在当前运行时镜像中的所有类和资源文件的整个集合。
所以,我认为获得的路径看起来没问题。我错在哪里了吗?
URI
转换为Path
的问题,还是modules
目录本来就应该对 jrtfs 的用户透明的? - Slaw