【发布时间】:2014-06-13 00:02:09
【问题描述】:
我想我会使用 Java 8 中的新 ResourceBundleControlProvider 框架来修复 Oracle 自己永远无法修复的问题 - 读取资源包时使用的默认编码。
于是我做了一个控件:
package com.acme.resources;
import java.io.IOException;
import java.util.Locale;
import java.util.ResourceBundle;
public class AcmeResourceBundleControl extends ResourceBundle.Control
{
@Override
public ResourceBundle newBundle(String baseName, Locale locale, String format,
ClassLoader loader, boolean reload)
throws IllegalAccessException, InstantiationException, IOException
{
throw new UnsupportedOperationException("TODO");
}
}
然后我做了一个提供者:
package com.acme.resources;
import java.util.ResourceBundle;
import java.util.spi.ResourceBundleControlProvider;
public class AcmeResourceBundleControlProvider implements ResourceBundleControlProvider
{
private static final ResourceBundle.Control CONTROL = new AcmeResourceBundleControl();
@Override
public ResourceBundle.Control getControl(String baseName)
{
if (baseName.startsWith("com.acme."))
{
return CONTROL;
}
else
{
return null;
}
}
}
然后在 META-INF/services/java.util.spi.ResourceBundleControlProvider:
com.acme.resources.AcmeResourceBundleControlProvider
然后我只是尝试从 IDEA 运行我们的应用程序,我发现它永远不会加载我的提供程序(否则会引发异常。)
我检查了名称,它们似乎都匹配。我检查了 IDEA 正在使用的编译器输出目录,它确实包含服务文件。我写了一个简单的测试程序,它只是试图查找服务:
public static void main(String[] args)
{
for (ResourceBundleControlProvider provider :
ServiceLoader.load(ResourceBundleControlProvider.class))
{
System.out.println(provider.getClass());
}
}
这会打印出一个条目,它是我的实现类的名称。所以问题是不是在服务文件中。
如果我在 ResourceBundle 中设置断点,我似乎能够访问自定义提供程序类。对调试器的初步尝试表明 ServiceLoader 没有找到任何实现,但我不知道为什么。我敢肯定有一些狡猾的类加载器魔法会导致不加载我的类。 :(
the Javadoc 上的一些可怕文档听起来好像必须作为全局扩展安装。如果真是这样,那就有点可惜了,因为这似乎是一种有用的方法来覆盖默认(在我看来是被破坏的)行为。但我也阅读了the tutorial 关于此事的内容,它似乎没有描述任何类似的内容(除非在最后一分钟从 Java 8 中删除了良好的行为并且文档已过时!)
【问题讨论】:
-
如果您手动 jar 应用程序然后运行 jar(确保目录结构)会发生什么?我不是 IDEA 用户,但我只是在想
META-INF/services/java.util.spi.ResourceBundleControlProvider可能不在运行时类路径上,或者它没有被检查。 -
从 jar 运行似乎也没有调用我的提供程序。所以很奇怪,my 代码中的 ServiceLoader.load() 找到了提供者,但 ResourceBundle.getBundle() 没有,这似乎完全违背了拥有提供者框架的目的。 ://
标签: java internationalization resourcebundle