国际化 Java 应用程序

编写:徐建祥(netpirate@gmail.com)

时间:2006-08-18 03:42

来自:http://www.anymobile.org

 

<术语>
国际化:internationalize,简称i18n,是指应用程序可以适应不同语言和地区的变化而变化的方案。
本地化:localization,检查l10n,是指应用程序适应某特定语言和地区的过程。

软件国际化是软件发展之必然,几乎所有的大众软件都支持多语言,最常见的就是英文软件的汉化版。

Java语言提供了基于unicode的国际化支持。通过Locale对象定义地区、语言、操作系统等信息,调用ResourceBundle的getBundle方法实现资源文件的数据绑定,轻松本地化不同的语言版本。

通过 ResourceBundle 和 Locale 2个类的定义可以看出 java 读取资源的流程为:language, country, variant -n-> language, country -n-> language -n-> Locale.getDefault()。

ResourceBundle.getBundle(String baseName)
ResourceBundle.getBundle(String baseName, Locale locale)

Locale(String language)
Locale(String language, String country)
Locale(String language, String country, String variant)

ResourceBundle 储存了键-值对形式的本地特殊化对象,如String,分为2种:ListResourceBundle和PropertyResourceBundle。

前者通过.class程序实现,后者通过.properties配置文件实现本地化,样例如下:

1、ListResourceBundle Example

1.1 默认(测试环境为中文)资源类

/**
 * $File: Rb.java             $ 
 * $Revision: 1.0.0           $
 * $ $Date: 2006-8-18 2:12:11 $
 */

package  org.anymobile.test;

import  java.util.ListResourceBundle;

/**
 * 
 * @author Xu Jianxiang
 */

public   class  Rb  extends  ListResourceBundle
{
    static final Object[][] contents = {
            { "CHG", "修改" }
            { "MSG", "消息" }
            { "BYE", "再见" }
    }
;

    public Object[][] getContents()
    {
        return contents;
    }

}

1.2 英文资源类

/**
 * $File: Rb_en.java        $ 
 * $Revision: 1.0.0         $ 
 * $Date: 2006-8-18 2:14:37 $
 */

package  org.anymobile.test;

import  java.util.ListResourceBundle;

/**
 * 
 * @author Xu Jianxiang
 */

public   class  Rb_en  extends  ListResourceBundle
{
    static final Object[][] contents = {
            { "CHG", "change" },
            { "MSG", "message" },
            { "BYE", "bye" }
    }
;

    public Object[][] getContents()
    {
        return contents;
    }

}

1.3 测试类

/**
 * $File: MyListResourceBundle.java $
 * $Revision: 1.0.0                 $
 * $Date: 2006-8-18 2:24:30         $
 */

package  org.anymobile.test;

import  java.util.Locale;
import  java.util.MissingResourceException;
import  java.util.ResourceBundle;

/**
 * Test java.util.ListResourceBundle
 * 
 * @author Xu Jianxiang
 */

public   class  MyListResourceBundle
{
    public static void main( String[] args )
    {
        String baseName = "org.anymobile.test.Rb";
        String key = "BYE";
        try
        {
            ResourceBundle res = ResourceBundle.getBundle( baseName );
            System.out.println( res.getString( key ) );
            
            res = ResourceBundle.getBundle( baseName, Locale.ENGLISH );
            System.out.println( res.getString( key ) );
        }

        catch ( MissingResourceException exp )
        {
            exp.printStackTrace();
        }

    }

}

运行结果:

再见
bye

2、PropertyResourceBundle Example

2.1 英文配置文件({project}/classes/resources/rb_en.properties)

# rb_en.properties

CHG = change
MSG = message
BYE = bye

2.2 中文配置文件({project}/classes/resources/rb_zn_CN.properties)

# rb_zn_CN.properties

CHG = 修改
MSG = 消息
BYE = 再见

2.3 测试类

/**
 * $File: MyResourceBoundle.java $
 * $Revision: 1.0.0              $ 
 * $Date: 2006-8-18 1:19:06      $
 */

package  org.anymobile.test;

import  java.util.Locale;
import  java.util.MissingResourceException;
import  java.util.ResourceBundle;

/**
 * Test java.util.PropertyResourceBundle
 * 
 * @author Xu Jianxiang
 */

public   class  MyPropResourceBundle
{
    public static void main( String[] args )
    {
        String baseName = "resources.rb";
        String key = "CHG";

        try
        {
//            ResourceBundle res = ResourceBundle.getBundle( baseName, Locale
//                    .getDefault() );
            ResourceBundle res = ResourceBundle.getBundle( baseName, new Locale("zh","CN","WINDOWS") );
            System.out.println( key + ": " + res.getString( key ) );

            res = ResourceBundle.getBundle( baseName, Locale.ENGLISH );
            System.out.println( key + ": " + res.getString( key ) );
        }

        catch ( MissingResourceException exp )
        {
            exp.printStackTrace();
        }

    }

}

运行结果:

CHG: 改变
CHG: change


另外,为了不致于在程序中产生乱码,可以通过jdk自带的native2ascii程序将.properties文件从本地编码转换成unicode编号格式的数据文件,为避免读取时因编码问题而出现乱码,可以先编写一个src.txt文件,然后执行指令:native2ascii src.txt dest.properties。转换后,如“中国”的unicode字符码为“/u4e2d/u56fd”。
如果需要将unicode恢复成原编码,可以执行指令:native2ascii -reverse dest.file src.file。

每次必须运行native2ascii方法比较繁琐,实际开发中,可以通过Apache Ant的native2Ascii任务进行批量转码。如:<native2ascii encoding="GBK" src="${src}" dest="${build}"/>

3、Spring 的消息国际化

org.springframework.context.ApplicationContext继承了org.springframework.context.MessgeResource接口,通过起getMessage()的各个版本读取文字消息的资源文件,从而实现消息国际化。

public interface MessageSource
{
String getMessage(String code, Object[] args, String defaultMessage, Locale locale);
}

参考资料:

Java 2 SDK Document
How to internationalize Java applications(Sun) http://java.sun.com/docs/books/tutorial/i18n/index.html
ISO Language Code    http://www.ics.uci.edu/pub/ietf/http/related/iso639.txt
ISO Country Code      http://www.chemie.fu-berlin.de/diverse/doc/ISO_3166.html

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值