国际化是指应用程序运行时,可根据客户端请求来自的国家/地区、语言的不同而显示不同的界面。
Java语言内核基于Unicode2.1,提供了对不同国家和不同语言文字的内部支持。
国际化的英文单词是Internationlization,因为单词过长,简称I18n。
2.1 首先,生成中文过渡文件:message_zh_CN_org.properties
2.2 其次,使用native2ascii.exe命令来生成message_zh_CN.properties文件
2.3 生成英文message_en_US.properties
3. JSP页面的国际化(register_i18n.jsp)
4. 生成Action类(RegisterI18nAction.java)
5. 国际化验证文件(RegisterI18nAction-validation.xml)
一:国际化资源文件加载优先顺序(Action类,调用getText("login")方法)
国际化的基本步骤
1. 修改struts.xml文件,以加载全局资源文件
<constant name="struts.custom.i18n.resources" value="message"></constant> <constant name="struts.i18n.encoding" value="UTF-8"></constant>
上述代码建议放在struts.xml文件的开头位置。全局资源文件名以“message_区域名.properties“格式命名。
2. 创建不同语言的资源文件
为了方便,中文的资源文件建议以message_zh_CN_org.properties命名(UTF-8文件格式),然后用native2ascii工具将它转换为message_zh_CN.properties文件供项目使用。
2.1 首先,生成中文过渡文件:message_zh_CN_org.properties
title=用户注册 name=用户名 username=真实姓名 pass=密码 repass=重输密码 sex=性别 province=省 age=年龄 birth=生日 love=爱好 mobile=手机 email=电子邮件 submit=提交 male=男 famale=女 chongqing=重庆 beijing=北京 shanghai=上海 tianjin=天津 swim=游泳 walk=散步 playtabletennis=乒乓球 reading=读书 others=其它 validate_name_null=请输入用户名 validate_name_scope=用户名必须在6到18位之间 validate_pass_null=请输入密码 validate_pass_scope=密码必须在6到12位之间,且只能是字母或数字 validate_repass_null=请输入重复密码 validate_repass_scope=两次输入的密码必须一致 validate_age_scope=年龄必须在 ${min} 和 ${max}之间 validate_birth_scope=生日必须在 ${min} 和 ${max}之间 validate_email_scope=电子邮件地址无效
以UTF-8格式保存此文件(如果是用UrltraEdit编辑,菜单:文件-》转换-》ASCII到UTF-8)。
2.2 其次,使用native2ascii.exe命令来生成message_zh_CN.properties文件
native2ascii -encodeing utf-8 message_zh_CN_org.properties message_zh_CN.properties
注意:
- native2ascii.exe文件位于javain目录下;
- message_zh_CN_org.properties文件必须是以UTF-8格式保存,这样上述参数“utf-8”才有意义。
最后生成文件如下:
title=u7528u6237u6ce8u518c name=u7528u6237u540d username=u771fu5b9eu59d3u540d pass=u5bc6u7801 repass=u91cdu8f93u5bc6u7801 sex=u6027u522b province=u7701 age=u5e74u9f84 birth=u751fu65e5 love=u7231u597d mobile=u624bu673a email=u7535u5b50u90aeu4ef6 submit=u63d0u4ea4 male=u7537 famale=u5973 chongqing=u91cdu5e86 beijing=u5317u4eac shanghai=u4e0au6d77 tianjin=u5929u6d25 swim=u6e38u6cf3 walk=u6563u6b65 playtabletennis=u4e52u4e53u7403 reading=u8bfbu4e66 others=u5176u5b83 validate_name_null=u8bf7u8f93u5165u7528u6237u540d validate_name_scope=u7528u6237u540du5fc5u987bu57286u523018u4f4du4e4bu95f4 validate_pass_null=u8bf7u8f93u5165u5bc6u7801 validate_pass_scope=u5bc6u7801u5fc5u987bu57286u523012u4f4du4e4bu95f4uff0cu4e14u53eau80fdu662fu5b57u6bcdu6216u6570u5b57 validate_repass_null=u8bf7u8f93u5165u91cdu590du5bc6u7801 validate_repass_scope=u4e24u6b21u8f93u5165u7684u5bc6u7801u5fc5u987bu4e00u81f4 validate_age_scope=u5e74u9f84u5fc5u987bu5728 ${min} u548c ${max}u4e4bu95f4 validate_birth_scope=u751fu65e5u5fc5u987bu5728 ${min} u548c ${max}u4e4bu95f4 validate_email_scope=u7535u5b50u90aeu4ef6u5730u5740u65e0u6548
2.3 生成英文message_en_US.properties
title=USER REGISTER name=member name username=real name pass=input password repass=confirm password sex=sex province=province age=age birth=birth love=love mobile=mobile email=email submit=submit male=male famale=famale chongqing=chongqing beijing=beijing shanghai=shanghai tianjin=tianjin swim=swim walk=walk playtabletennis=playtabletennis reading=reading others=others validate_name_null=please input member name validate_name_scope=member name must be between 6 and 18 validate_pass_null=please input password validate_pass_scope=password must be between 6 and 12 and only used number or letter validate_repass_null=please input confirm password validate_repass_scope=two password must be same validate_age_scope=age must be between ${min}and ${max} validate_birth_scope=birthday must be between ${min}and ${max} validate_email_scope=email is invalid
3. JSP页面的国际化(register_i18n.jsp)
在页面中读取国际化信息有三种方式:
3.1 使用s:text标签
<s:text name="title"></s:text>
3.2 在页面表单中的各个输入标签中加入key属性
<s:textfield name="username" key="username"></s:textfield>
3.3 使用getText()表达式
<s:radio list="#{'1':getText('male'),'0':getText('famale')}" value="1" name="sex" key="sex"></s:radio>
下面是国际化后的代码:
<%@ page language="java" import="java.util.*" pageEncoding="utf-8"%> <%@ page isELIgnored="false"%> <%@ taglib uri="/struts-tags" prefix="s"%> <%@ taglib uri="/struts-dojo-tags" prefix="sx"%> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN"> <html> <head> <sx:head extraLocales="utf-8"/> <title><s:text name="title"></s:text></title> </head> <body> <s:form action="register_i18n" method="post" theme="xhtml"> <s:textfield name="name" key="name"></s:textfield> <s:textfield name="username" key="username"></s:textfield> <s:textfield name="pass" key="pass"></s:textfield> <s:textfield name="repass" key="repass"></s:textfield> <s:radio list="#{'1':getText('male'),'0':getText('famale')}" value="1" name="sex" key="sex"></s:radio> <s:select name="province" list="#{'0':getText('chongqing'),'1':getText('beijing'),'2':getText('shanghai'),'3':getText('tianjin')}" key="province"></s:select> <s:textfield name="age" key="age"></s:textfield> <sx:datetimepicker name="birth" displayFormat="yyyy-MM-dd" key="birth" accesskey="false"></sx:datetimepicker> <s:checkboxlist name="love" key="love" list="#{'0':getText('swim'),'1':getText('walk'),'2':getText('playtabletennis'),'3':getText('reading'),'4':getText('others')}"></s:checkboxlist> <s:textfield name="mobile" key="mobile"></s:textfield> <s:textfield name="email" key="email"></s:textfield> <s:textfield name="request_locale" key="request_locale"></s:textfield> <s:submit key="submit"></s:submit> </s:form> </body> </html>
本演示采用了xhtml主题,在form标签中有“theme=xhtml“代码。
4. 生成Action类(RegisterI18nAction.java)
package com.clzhang.struts2.demo1; import java.util.*; import com.opensymphony.xwork2.ActionSupport; public class RegisterI18nAction extends ActionSupport { public static final long serialVersionUID = 1; private String name; private String username; private String pass; private String repass; private String sex; private String province; private Integer age; private Date birth; private String love; private String mobile; private String email; public String getName() { return name; } public void setName(String name) { this.name = name; } public String getUsername() { return username; } public void setUsername(String username) { this.username = username; } public String getPass() { return pass; } public void setPass(String pass) { this.pass = pass; } public String getRepass() { return repass; } public void setRepass(String repass) { this.repass = repass; } public String getSex() { return sex; } public void setSex(String sex) { this.sex = sex; } public String getProvince() { return province; } public void setProvince(String province) { this.province = province; } public Integer getAge() { return age; } public void setAge(Integer age) { this.age = age; } public Date getBirth() { return birth; } public void setBirth(Date birth) { this.birth = birth; } public String getLove() { return love; } public void setLove(String love) { this.love = love; } public String getMobile() { return mobile; } public void setMobile(String mobile) { this.mobile = mobile; } public String getEmail() { return email; } public void setEmail(String email) { this.email = email; } @Override public String execute() { System.out.println(username + "|" + age + "|" + mobile + " register(i18n) finished!"); return SUCCESS; } }
其实这个Action类只有相关setter/getter而已,此演示我们并不需要它真正实现什么业务功能的。
5. 国际化验证文件(RegisterI18nAction-validation.xml)
<!DOCTYPE validators PUBLIC "-//Apache Struts//XWork Validator 1.0.2//EN" "http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd"> <validators> <field name="name"> <field-validator type="requiredstring"> <param name="trim">true</param> <message key="validate_name_null"></message> </field-validator> <field-validator type="stringlength"> <param name="minLength">6</param> <param name="maxLength">18</param> <message key="validate_name_scope"></message> </field-validator> </field> <field name="pass"> <field-validator type="requiredstring"> <param name="trim">true</param> <message key="validate_pass_null"></message> </field-validator> <field-validator type="regex"> <param name="expression"><![CDATA[w{6,12}]]></param> <message key="validate_pass_scope"></message> </field-validator> </field> <field name="repass"> <field-validator type="requiredstring"> <param name="trim">true</param> <message key="validate_repass_null"></message> </field-validator> <field-validator type="fieldexpression"> <param name="expression"><![CDATA[repass==pass]]></param> <!--这里也可以用repass.equals(pass)//--> <message key="validate_repass_scope"></message> </field-validator> </field> <field name="age"> <field-validator type="int"> <param name="min">1</param> <param name="max">150</param> <message key="validate_age_scope"></message> </field-validator> </field> <field name="birth"> <field-validator type="date"> <param name="min">1900-01-01</param> <param name="max">2050-01-01</param> <message key="validate_birth_scope"></message> </field-validator> </field> <field name="email"> <field-validator type="email"> <message key="validate_email_scope"></message> </field-validator> </field> </validators>
对此文件有不明白的地方,可以参考本文前些章节,以及文章:struts2:数据校验,validation.xml格式验证
6. 修改struts.xml文件,加入action配置
加入如下内容:
<action name="register_i18n" class="com.clzhang.struts2.demo1.RegisterI18nAction"> <result name="success">/struts2/demo1/success.jsp</result> <result name="input">/struts2/demo1/register_i18n.jsp</result> </action>
7. 测试
打开IE,输入地址:http://127.0.0.1:8080/st/ssh/demo1/register_i18n.jsp
结果如下:
直接提交,结果如下:
改变本机语言为英文,测试结果为:
注意:修改本机语言的方法,参考附录:二:临时修改用户默认语言环境 。
附录
一:国际化资源文件加载优先顺序(Action类,调用getText("login")方法)
a. 首先加载action同目录下且baseName为action类名的系列资源文件。
b. 如果在a中找不到指定key对应的消息,且action有父类,则加载父类同目录下baseName为父类名的系列资源文件。
c. 如果在b中找不到key对应的消息,且action有实现接口,则加载其实现的接口同目录下baseName为接口名的系列资源文件。
d. 如果在c中找不到key对应的消息,则查找当前包下baseName为包名的系列资源文件。
e. 如果在d中找不到key对应的消息,则沿着当次包上溯,直到最顶层包来查找baseName为包名的系列资源文件。
f. 如果e中找不到key对应的消息,则查找struts.custom.i18n.resources常量指定baseName的系列资源文件。
g. 经过上面的步骤还是找不到key对应的消息,将直接输出该key的字符串值;如果上面的任何一步找到对应的key的消息,系统停止搜索,直接输出该key。
二:国际化资源文件加载优先顺序(JSP文件)
如果<s:text.../>标签、表单标签没有使用<s:i18n.../>标签作为父标签,其加载顺序为:
直接加载struts.custom.i18n.resources常量指定baseName的全局范围国际化资源文件。如果找不到该key对应的value值,将直接输出该key的字符串值。
三:临时修改用户默认语言环境
方式一:
在“控制面板”-》“区域和语言”中将机器语言环境设置成相应的国家即可。
方式二:
向action额外提交一个request_locale的参数,取值为语言值即可。比如:request_locale=en_US、request_locale=zh_CN等。对于本演示而言,可以直接在request_locale文本框中输入相关值即可,如:en_US/zh_CN等。
因为struts2提供了一个名为i18n的拦截器,将其默认注册在拦截器栈中。i18n拦截器在执行action方法之前,自动查找请求中名为request_locale的参数。如果该参数存在,拦截器就将其作为参数,转换为Locale对象,并将其设为用户默认的Locale(代表国家、语言环境)。