国际化
国际化是指应用程序运行时,可根据客户端请求来自的国家/地区、语言的不同而显示不同的界面。例如,如果请求来自于中文操作系统的客户端,则应用程序中的各种提示信息错误和帮助等都使用中文文字;如果客户端使用英文操作系统,则应用程序能自动识别,并做出英文的响应。
国际化的英文单词是Internationalization,因为这个单词太长了,有时也简称I18N,其中I是这个单词的第一个字母,18表示中间省略的字母个数,而N代表这个单词的最后一个字母。
一个国际化支持很好的应用,在不同的区域使用时,会呈现出本地语言的提示。这个过程也被称为Localization,即本地化。类似于国际化可以称为I18N,本地化也可以称为L10N。
Java国际化的思路
Java程序的国际化思路是将程序中的标签、提示等信息放在资源文件中,程序需要支持哪些国家、语言环境,就对应提供相应的资源文件。资源文件是key-value对,每个资源文件中的key是不变的,但value则随不同的国家、语言而改变。
Java程序的国际化主要通过如下三个类完成。
- java.util.ResourceBundle:用于加载国家、语言资源包。
- java.util.Locale:用于封装特定的国家/区域、语言环境。
- 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程序国际化的关键类是ResourceBundle
和Locale
,ResourceBundle
根据不同的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文档说明