【问题标题】:Why isn't my ResourceBundleControlProvider being loaded?为什么我的 ResourceBundleControlProvider 没有被加载?
【发布时间】: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


【解决方案1】:

教程确实声明包含ResourceBundleControlProvider 的JAR 必须位于JVM 的系统扩展目录中。 tutorial 的第 6 节描述了要求:

java -Djava.ext.dirs=lib -cp build RBCPTest

安装 Java 扩展时,通常会将扩展的 JAR 文件放在 JRE 的 lib/ext 目录中。但是,此命令使用系统属性 java.ext.dirs 指定包含 Java 扩展的目录。

ServiceLoader.loadInstalled() 的 JavaDoc 还声明应用程序类路径上的提供程序被忽略。

【讨论】:

  • 是的,看起来是这样。因此,我们急切等待的另一个功能已经瘫痪到没有人可以使用它的地步。真好。
【解决方案2】:

您的问题是 JVM 附带的 java.util.ResourceBundle 执行 ServiceLoader.loadInstalled(ResourceBundleControlProvider.class) 以获取静态初始化程序中的提供者列表,然后使用由此获得的列表。

【讨论】:

    猜你喜欢
    • 2013-02-28
    • 2018-01-08
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2015-12-24
    • 2012-01-09
    • 1970-01-01
    • 2016-09-16
    相关资源
    最近更新 更多