一、国际化的概念
软件开发时,要使它能同时应对世界不同地区和国家的访问,并针对不同地区和国家的访问,提供相应的、符合来访者阅读习惯的页面或数据。国际化又称为 i18n:internationalization。软件实现国际化,需具备哪些特征:
- 对于程序中固定使用的文本元素,例如菜单栏、导航条等中使用的文本元素、或错误提示信息,状态信息等,需要根据来访者的地区和国家,选择不同语言的文本为之服务。
- 对于程序动态产生的数据,例如(日期,货币等),软件应能根据当前所在的国家或地区的文化习惯进行显示。
二、固定文本国际化
1、介绍
对于软件中的菜单栏、导航条、错误提示信息,状态信息等这些固定不变的文本信息,可以把它们写在一个properties文件中,并根据不同的国家编写不同的properties文件。这一组properties文件称之为一个资源包。在JavaAPI中提供了一个ResourceBundle 类用于描述一个资源包,并且 ResourceBundle类提供了相应的方法getBundle,这个方法可以根据来访者的国家地区自动获取与之对应的资源文件予以显示。
2、编写资源包
1.建立资源包
一个资源包中的每个资源文件都必须拥有共同的基名。除了基名,每个资源文件的名称中还必须有标识其本地信息的附加部分。
例如:
建立一个包(com.shen.resource)来存放资源文件,在该包下建立文件myproperites_zh.properties、myproperites_en.properties
注意:"_"后部分(zh和en)要查每个国家对应的值。
2.建立默认资源包
每个资源包都应有一个默认资源文件,这个文件不带有标识本地信息的附加部分。若ResourceBundle对象在资源包中找不到与用户匹配的资源文件,它将选择该资源包中与用户最相近的资源文件,如果再找不到,则使用默认资源文件。
例如:
在包(com.shen.resource)中创建文件myproperites.properties
3.编写资源文件
资源文件的内容通常采用“关键字=值”的形式,软件根据关键字检索值显示在页面上。一个资源包中的所有资源文件的关键字必须相同,值则为相应国家的文字。并且资源文件中采用的是properties格式文件,所以文件中的所有字符都必须是ASCII字码,对于像中文这样的非ACSII字符,须先进行编码。(java提供了一个native2ascII命令用于编码)。
例:
编写myproperites.properties:
username=username
password=password
submit=submit
编写myproperites_zh.properties:
username=\u7528\u6237\u540D
password=\u5BC6\u7801
submit=\u63D0\u4EA4
编写myproperites_en.properties:
username=username
password=password
submit=submit
注意:在jdk中bin目录下有native2ascII.exe可以用于将中文转为ASCII,调用方法为:打开cmd-》输入native2ascII即可启动,如之后输入”用户名“,将转为”\u7528\u6237\u540D“【粘贴方式为:选中-》鼠标右键即可】。myeclipes中在可视化中可以直接输入中文,它将自动转换。
2、调用资源包
ResourceBundle类提供了一个静态方法getBundle,该方法用于装载资源文件,并创建ResourceBundle实例。
例:
在包(com.shen.i18n)中建立程序Demo1.java
import java.util.Locale;
import java.util.ResourceBundle;
public class Demo1 {
/**
* @param args
*/
public static void main(String[] args) {
//注意该处为默认资源包的名
ResourceBundle bundle = ResourceBundle.getBundle("com.shen.resource.myproperties",Locale.US);
String username = bundle.getString("username");
String password= bundle.getString("password");
System.out.println(username);
System.out.println(password);
}
}
说明:在JavaAPI中提供了一个ResourceBundle 类用于描述一个资源包,并且 ResourceBundle类提供了相应的方法getBundle,这个方法可以根据来访者的国家地区自动获取与之对应的资源文件予以显示。资源包基名(且必须为完整路径)。如果与该locale对象匹配的资源包子类找不到。一般情况下,则选用默认资源文件予以显示。
三、动态数据国际化
1、介绍
数值,货币,时间,日期等数据由于可能在程序运行时动态产生,所以无法像文字一样简单地将它们从应用程序中分离出来,而是需要特殊处理。Java 中提供了解决这些问题的 API 类(位于 java.util 包和 java.text 包中)
2、日期国际化
1.DateFormat类介绍
DateFormat 类可以将一个日期/时间对象格式化为表示某个国家地区的日期/时间字符串。DateFormat 类除了可按国家地区格式化输出日期外,它还定义了一些用于描述日期/时间的显示模式的 int 型的常量,包括FULL, LONG, MEDIUM, DEFAULT, SHORT,实例化DateFormat对象时,可以使用这些常量,控制日期/时间的显示长度。
例如:
对于日期/时间“2005年9月15日下午4时41分20秒”,如设置了DateFormat的显示模式,则日期/时间显示样式分别为:
模式 | 中文环境 | 英文环境 |
SHORT | 05-9-15 下午4:41 | 9/15/05 4:41 PM |
MEDIUM | 2005-9-15 16:41:20 | Sep 15, 2005 4:41:20 PM |
LONG | 2005年9月15日 下午04时41分20秒 | September 15, 2005 4:41:20 PM CST |
FULL | 2005年9月15日 星期四 下午04时41分20秒 CST | Thursday, September 15, 2005 4:41:20 PM CST |
2.实例化 DateFormat
实例化DateFormat类有九种方式,以下三种为带参形式,下面列出的三种方式也可以分别不带参,或只带显示样式的参数。
- getDateInstance(int style, Locale aLocale):以指定的日期显示模式和本地信息来获得DateFormat实例对象,该实例对象不处理时间值部分。
- getTimeInstance(int style, Locale aLocale):以指定的时间显示模式和本地信息来获得DateFormat实例对象,该实例对象不处理日期值部分。
- getDateTimeInstance(int dateStyle, int timeStyle, Locale aLocale):以单独指定的日期显示模式、时间显示模式和本地信息来获得DateFormat实例对象。
3.DateFormat 对象的方法
DateFormat 对象的方法:
- format: 将date对象格式化为符合某个本地环境习惯的字符串。
- parse:将字符串解析为日期/时间对象
注意:
parse和format完全相反,一个是把date时间转化为相应地区和国家的显示样式,一个是把相应地区的时间日期转化成date对象,该方法在使用时,解析的时间或日期要符合指定的国家、地区格式,否则会抛异常。
DateFormat 对象通常不是线程安全的,每个线程都应该创建自己的 DateFormat 实例对象
4.程序演示
例:
在包(com.shen.i18n)中建立程序Demo2.java
import java.text.DateFormat;
import java.text.ParseException;
import java.util.Date;
import java.util.Locale;
public class Demo2 {
/**
* @param args
* @throws ParseException
*/
public static void main(String[] args) throws ParseException {
Date date = new Date(); //代表在这个new出来的这一刻的时间(日期、时间)
//输出日期部分
DateFormat df = DateFormat.getDateInstance(DateFormat.FULL, Locale.GERMAN);
String result = df.format(date);
System.out.println(result);
//输出时间部分
df = DateFormat.getTimeInstance(DateFormat.FULL, Locale.CHINA);
result = df.format(date);
System.out.println(result);
//输出日期和时间
df = DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.LONG, Locale.CHINA);
result = df.format(date);
System.out.println(result);
//把字符串反向解析成一个date对象
String s = "10-9-26 下午02时49分53秒";
df = DateFormat.getDateTimeInstance(DateFormat.SHORT, DateFormat.LONG, Locale.CHINA);
Date d = df.parse(s);
System.out.println(d);
}
}
2、数字格式化
1.NumberFormat类的介绍
NumberFormat 可以将一个数值格式化为符合某个国家地区习惯的数值字符串,也可以将符合某个国家地区习惯的数值字符串解析为对应的数值
2.NumberFormat类的实例化
实例化NumberFormat类时,可以使用locale对象作为参数,也可以不使用,下面列出的是使用参数的:
- getNumberInstance(Locale locale):以参数locale对象所标识的本地信息来获得具有多种用途的NumberFormat实例对象
- getIntegerInstance(Locale locale):以参数locale对象所标识的本地信息来获得处理整数的NumberFormat实例对象
- getCurrencyInstance(Locale locale):以参数locale对象所标识的本地信息来获得处理货币的NumberFormat实例对象
- getPercentInstance(Locale locale):以参数locale对象所标识的本地信息来获得处理百分比数值的NumberFormat实例对象
3.NumberFormat类的方法
- format 方法:将一个数值格式化为符合某个国家地区习惯的数值字符串
- parse 方法:将符合某个国家地区习惯的数值字符串解析为对应的数值。
4.程序演示
public class Demo3 {
/**
* NumberFormat实例
* @throws ParseException
*/
public static void main(String[] args) throws ParseException {
int price = 89;
NumberFormat nf = NumberFormat.getCurrencyInstance(Locale.CHINA);
String result = nf.format(price);
System.out.println(result);//¥89.0
String s = "¥89.00";//将"¥89.00"转为数字
nf = NumberFormat.getCurrencyInstance(Locale.CHINA);
Number n = nf.parse(s);
System.out.println(n.doubleValue()+1);//90.0
double num = 0.5;//将其转换为50%
nf = NumberFormat.getPercentInstance();
System.out.println(nf.format(num));
}
}
四、批量国际化
1、模式字符串和占位符
有字符串mess=“At 12:30 pm on jul 3,1998, a hurricance destroyed 99 houses and caused $1000000 of damage”
占位符即为:{0}、{1}.......
模式字符串即为:At {0} on {1},a destroyed {2} houses and caused {3} of damage
补充:
占位符有三种方式书写方式:
- {argumentIndex}: 0-9 之间的数字,表示要格式化对象数据在参数数组中的索引号
- {argumentIndex,formatType}: 参数的格式化类型
- {argumentIndex,formatType,FormatStyle}: 格式化的样式,它的值必须是与格式化类型相匹配的合法模式、或表示合法模式的字符串。
2、MessageFormat类的使用
MessageFormat(String pattern)
实例化MessageFormat对象,并装载相应的模式字符串。
format(object obj[])
格式化输出模式字符串,参数数组中指定占位符相应的替换对象。
String pattern = "On {0}, a hurricance destroyed {1} houses and caused {2} of damage.";
MessageFormat format = new MessageFormat(pattern,Locale.CHINA);
Object arr[] = {new Date(),99,100000000};
String result = format.format(arr);
System.out.println(result);
3、实际应用
在myproperites_zh.properties中:
message=\u5728{0}\uFF0C\u4E00\u573A\u60E7\u98CE\uFF0C\u6467\u6BC1\u4E86{1}\u6240\u623F\u5B50\uFF0C\u5BFC\u81F4{2}\u94B1\u7684\u635F\u5931
在myproperites_(en).properties中:
message=At {0, time, short} on {0, date}, a destroyed {1} houses and caused {2, number, currency} of damage.
说明:{0,time,short}表示显示时间,类型为短......
在程序中:
ResourceBundle bundle = ResourceBundle.getBundle("com.shen.resource.myproperties",Locale.CHINA);
pattern = bundle.getString("message");
format = new MessageFormat(pattern,Locale.CHINA);
arr = new Object[]{new java.util.Date(),99,10000000};
result = format.format(arr);
System.out.println(result);