java - 为什么没有加载我的 ResourceBundleControlProvider?

标签 java internationalization resourcebundle

我想我会使用 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 包含 ResourceBundleControlProvider必须在 JVM 的系统扩展目录中。 tutorial 的第 6 节描述要求:

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

When you install a Java extension, you typically put the JAR file of the extension in the lib/ext directory of your JRE. However, this command specifies the directory that contains Java extensions with the system property java.ext.dirs.

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

关于java - 为什么没有加载我的 ResourceBundleControlProvider?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24195777/

相关文章:

javascript - JavaScript 中的 JSF 资源包参数?

java - 使用资源包更改标签

JAVAFX 鼠标点击某个值

java - 从禁用 JavaScript 的 htmlunit WebClient 调用 getPage 并将 setTimeout 设置为 10000 永远等待

cakephp - 如何本地化 CakePHP 插件?

python - Django 模板中的编码问题

java - 基于插件的程序中的语言环境和 ResourceBundle

java - RMI stub 和子类继承

java - 无法使用 Ubuntu Oracle Java 8 运行 IntelliJ IDEA CE 12

database - 标记内容系统 - 使用 i18n