国际化与格式化

国际化

国际化是指应用程序运行时,可根据客户端请求来自的国家/地区、语言的不同而显示不同的界面。例如,如果请求来自于中文操作系统的客户端,则应用程序中的各种提示信息错误和帮助等都使用中文文字;如果客户端使用英文操作系统,则应用程序能自动识别,并做出英文的响应。

国际化的英文单词是Internationalization,因为这个单词太长了,有时也简称I18N,其中I是这个单词的第一个字母,18表示中间省略的字母个数,而N代表这个单词的最后一个字母。

一个国际化支持很好的应用,在不同的区域使用时,会呈现出本地语言的提示。这个过程也被称为Localization,即本地化。类似于国际化可以称为I18N,本地化也可以称为L10N。

Java国际化的思路

Java程序的国际化思路是将程序中的标签、提示等信息放在资源文件中,程序需要支持哪些国家、语言环境,就对应提供相应的资源文件。资源文件是key-value对,每个资源文件中的key是不变的,但value则随不同的国家、语言而改变。

Java程序的国际化主要通过如下三个类完成。

  1. java.util.ResourceBundle:用于加载国家、语言资源包。
  2. java.util.Locale:用于封装特定的国家/区域、语言环境。
  3. java.text.MessageFormat:用于格式化带占位符的字符串。

为了实现程序的国际化,必须先提供程序所需要的资源文件,资源文件的命名可以有如下三种形式。

1)baseName_language_country.properties
(2)baseName_language.properties
(3)baseName.properties

其中baseName是资源文件的基本名,可随意指定;而language和country都不可随意变化,必须是Java所支持的语言和国家。

Java支持的国家和语言

事实上,Java不可能支持所有的国家和语言,如果需要获取Java所支持的国家和语言,则可调用Locale类的getAvailableLocales()方法,该方法返回一个Locale数组,该数组里包含了Java所支持的国家和语言。

public class LocaleList {

    public static void main(String[] args) {
        //返回Java所支持的全部国家和语言的数组
        Locale[] localList = Locale.getAvailableLocales();

        for (int i=0; i<localList.length; i++){
            System.out.println(localList[i].getDisplayCountry()
                    + "=" + localList[i].getCountry() + " "
                    + localList[i].getDisplayLanguage() + "=" + localList[i].getLanguage());
        }

    }

}

完成程序国际化

先准备好资源文件
mess_zh_CN.properties

hello=你好

mess_en_US.properties

hello=Welcome You

由于mess_zh_CN.properties文件中包含了中文,需要转换一下编码才不会导致输出时乱码,Java提供了一个工具来处理该文件:native2ascii,这个工具可以在%JAVA_HOME%/bin路径下找到。

native2ascii 资源源文件 目的资源文件

代码如下

public class Hello {

    public static void main(String[] args) {
        //取得系统默认的国家/语言环境
        Locale myLocale = Locale.getDefault();
        System.out.println(myLocale.getDisplayCountry() + "  " + myLocale.getCountry());
        System.out.println(myLocale.getDisplayLanguage() + "  " + myLocale.getLanguage());
        //根据指定的国家/语言环境加载资源文件
        ResourceBundle bundle = ResourceBundle.getBundle("mess", myLocale);
        //打印从资源文件中取得的消息
        System.out.println(bundle.getString("hello"));
    }

}

可以看到输出结果

中国  CN
中文  zh
你好

在这里插入图片描述
如果将系统中的语言环境设置成美国,输出结果将会是Welcome You字符串

从上面程序可以看出,如果希望程序完成国际化,只需要将不同的国家/语言的提示信息分别以不同的文件存放即可。例如,简体中文的语言资源文件就是xxx_zh_CN.properties文件,而美国英语的语言资源文件就是xxx_en_US.properties文件。

Java程序国际化的关键类是ResourceBundleLocaleResourceBundle根据不同的Locale加载语言资源文件,再根据指定的key取得已加载语言资源文件中的字符串。

使用MessageFormat处理包含占位符的字符串

如果需要输出的消息中必须包含动态的内容,则可以使用带占位符的消息。

msg=你好,{0}!今天是{1}

当我们直接使用ResourceBundle的getString()方法来取出msg对应的字符串时,在简体中文环境下得到你好,{0}!今天是{1}。字符串,这显然不是我们需要的结果,我们还应为{0}和{1}两个占位符赋值。此时需要使用MessageFormat类

    public static String format(String pattern, Object ... arguments) {
        MessageFormat temp = new MessageFormat(pattern);
        return temp.format(arguments);
    }

返回后面的多个参数值填充前面的pattern字符串,其中pattern字符串不是正则表达式,而是一个带占位符的字符串。

程序改造如下

public class Hello {

    public static void main(String[] args) {
        //取得系统默认的国家/语言环境
        Locale myLocale = Locale.getDefault();
        System.out.println(myLocale.getDisplayCountry() + "  " + myLocale.getCountry());
        System.out.println(myLocale.getDisplayLanguage() + "  " + myLocale.getLanguage());
        //根据指定的国家/语言环境加载资源文件
        ResourceBundle bundle = ResourceBundle.getBundle("mess", myLocale);
        String msg = bundle.getString("msg");
        //使用MessageFormat为带占位符的字符串传入参数
        System.out.println(MessageFormat.format(msg, "小明", new Date()));
    }

}
中国  CN
中文  zh
你好,小明!今天是20-6-8 上午11:16

使用SimpleDateFormat格式化日期

DateFormat的parse()方法可以把字符串解析成Date对象,但实际上DateFormat的parse()方法不够灵活,它要求被解析的字符串必须满足特定的格式,为了更好地格式化日期、解析日期字符串,Java提供了SimpleDateFormat类。

SimpleDateFormat是DateFormat的子类,正如它的名字所暗示的,它是简单的日期格式器。实际上SimpleDateFormat比DateFormat更简单,功能更强大。

SimpleDateFormat可以非常灵活地格式化Date,也可以用于解析各种格式的日期字符串。创建SimpleDateFormat对象时需要传入一个pattern字符串,这个pattern不是正则表达式,而是一个日期模板字符串。

public class SimpleDateFormatTest {

    public static void main(String[] args) throws Exception{
        Date d = new Date();
        //创建一个SimpleDateFormat对象
        SimpleDateFormat sdf1 = new SimpleDateFormat("Gyyyy年中第D天");
        //将d格式化成日期
        String dateStr = sdf1.format(d);
        System.out.println(dateStr);

        //一个非常特殊的日期字符串
        String str = "07###三月##21";
        SimpleDateFormat sdf2 = new SimpleDateFormat("y###MMM##d");
        System.out.println(sdf2.parse(str));

    }

}

输出结果如下

公元2020年中第160天
Wed Mar 21 00:00:00 CST 2007

从上面程序中可以看出,使用SimpleDateFormat可以将日期格式化成形如公元2020年中第160天这样的字符串,也可以把形如07###三月##21这样的字符串解析成日期,功能非常强大。SimpleDateFormat把日期格式化成怎样的字符串,以及能把怎样的字符串解析成Date,完全取决于创建该对象时指定的pattern参数,pattern是一个使用日期字段占位符的日期模板。

SimpleDateFormat支持哪些日期、时间占位符,可以自行查阅API文档说明

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章