假设java代码库有一个名为“com.example”的包。

在运行时,我们可以通过调用

来获取此包
Package p = Package.getPackage( "com.example" ); //(returns null)

甚至可以通过调用

获取所有包的列表
Packages[] ps = Package.getPackages();

问题是 - 如果ClassLoader尚未从包中加载任何类,则它们将无法用于这些函数调用。我们可以通过首先强制加载包中的一个类来强制它加载包,如下所示:

this.getClass().getClassLoader().loadClass( "com.example.SomeClass" );
Package p = Package.getPackage( "com.example" ); //(returns non-null)

然而,这很麻烦,需要提前知道属于该软件包的某些类的名称。

所以问题是 - 有没有办法按名称获取Package的实例,无论ClassLoader是否做了什么?关于类加载/包在这种情况下如何工作的假设是否准确?

有帮助吗?

解决方案

我认为你需要这个,因为你需要检查它的注释。否则,您不会对包引用感兴趣,只有操作才能访问注释。这导致了你也有一个package-info.java定义了一些注释。

如果你检查 java.lang.Package ,你会发现 getPackageInfo 只是将package-info类加载为普通类。

我遇到了同样的问题并提出了这个解决方案。

public static Package getPackage(String packageName) throws ClassNotFoundException {
    Class.forName(packageName+".package-info"); // makes sure package info exist and that the class loader already knows about the package
    return Package.getPackage(packageName);
}

其他提示

或者,您可以使用类根目录作为起点,并遍历所有* .class文件和子目录。只有在事先知道所有.class文件所在的位置时,这才有效。

所有这一切的原因是Java具有动态类加载,因此可以在运行时从编译时未知的位置或甚至在启动时加载类。因此,包的概念只是加载类的命名空间,而不是可用于查找它们的目录。

我担心你的假设是无效的。类加载器在加载类时会打包簿记。

您可以将通配符传递给 ClassLoader.getResources ,并强制它接收包中的类,这将依次完成工作。

您可以创建自己的ClassLoader来调用 definePackage ,但这对使用中常用的vanilla类加载器没有帮助。

许可以下: CC-BY-SA归因
不隶属于 StackOverflow
scroll top