如何避免Java ResourceBundle字符串中的重复?

Tre*_*kaz 14 java resourcebundle substitution

我们有很多字符串包含相同的子字符串,从关于检查日志或如何联系支持的句子到包含公司或产品名称的类似品牌的字符串.重复导致了我们自己的一些问题(主要是拼写错误或复制/粘贴错误),但它也会导致问题,因为它增加了翻译者必须翻译的文本数量.

我提出的解决方案是这样的:

public class ExpandingResourceBundleControl extends ResourceBundle.Control {
  public static final ResourceBundle.Control EXPANDING =
    new ExpandingResourceBundleControl();

  private ExpandingResourceBundleControl() { }

  @Override
  public ResourceBundle newBundle(String baseName, Locale locale, String format,
                                  ClassLoader loader, boolean reload)
    throws IllegalAccessException, InstantiationException, IOException {

      ResourceBundle inner = super.newBundle(baseName, locale, format, loader, reload);
      return inner == null ? null : new ExpandingResourceBundle(inner, loader);
  }
}
Run Code Online (Sandbox Code Playgroud)

ExpandingResourceBundle 委托实际资源包但执行{{this.kind.of.thing}}的转换以查找资源中的密钥.

每当你想要获得其中一个,你必须去:

ResourceBundle.getBundle("com/acme/app/Bundle", EXPANDING);
Run Code Online (Sandbox Code Playgroud)

这工作正常 - 一段时间.

最终发生的是一些新代码(在我们的例子中是自动生成的代码,它是由马蒂斯吐出来的)在不指定自定义控件的情况下查找相同的资源包.如果你编写一个简单的单元测试来调用它然后没有,这似乎是不可重现的,但它是在应用程序运行真实时发生的.不知何故,内部缓存ResourceBundle弹出好的值并用破坏的值替换它.我还没弄清楚为什么Sun的jar文件是在没有调试信息的情况下编译的,所以调试它是一件苦差事.

我的问题:

  1. 有没有办法全局设置我可能不知道的默认ResourceBundle.Control?这样可以相当优雅地解决所有问题.

  2. 有没有其他方式优雅地处理这种事情,也许根本没有篡改ResourceBundle类?

Mat*_*ock 6

我认为这是ResourceBundles设计运行方式的一个根本缺陷:引用其他键的键会自动违反DRY(不要重复自己)原则.我解决这个问题的方法类似于你的方法:创建一个ReflectiveResourceBundle类,它允许你使用EL表示法在消息中指定资源键.

错误的方法:

my.name.first=Bob
my.name.last=Smith
my.name.full=Bob Smith
Run Code Online (Sandbox Code Playgroud)

正确的方式:

my.name.first=Bob
my.name.last=Smith
my.name.full=${my.name.first} ${my.name.last}
Run Code Online (Sandbox Code Playgroud)

我已将代码上传到GitHub,以便您或其他任何人可以下载它.此外,我为使用Stripes Framework(http://www.stripesframework.org/)的任何人添加了一些示例代码,以帮助您快速启动并运行.

让它与标准JSTL fmt taglib一起工作的技巧是设置一个拦截器,用我们自己的资源替换HttpServletRequest的资源.代码看起来像这样:

ResourceBundle bundle = MyStaticResourceHoldingTheBundle.getBundle();
Config.set(request, Config.FMT_LOCALIZATION_CONTEXT, new LocalizationContext(bundle, locale));
Run Code Online (Sandbox Code Playgroud)

有关更多详细信息,请查看上面链接中的stripes.interceptor包.