struts2:国际化

国际化是指应用程序运行时,可根据客户端请求来自的国家/地区、语言的不同而显示不同的界面。

Java语言内核基于Unicode2.1,提供了对不同国家和不同语言文字的内部支持。

国际化的英文单词是Internationlization,因为单词过长,简称I18n。

国际化的基本步骤

1. 修改struts.xml文件,以加载全局资源文件

2. 创建不同语言的资源文件

  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)

  3.1 使用s:text标签

  3.2 在页面表单中的各个输入标签中加入key属性

  3.3 使用getText()表达式

4. 生成Action类(RegisterI18nAction.java)

5. 国际化验证文件(RegisterI18nAction-validation.xml)

6. 修改struts.xml文件,加入action配置

7. 测试

附录

一:国际化资源文件加载优先顺序(Action类,调用getText("login")方法)

二:国际化资源文件加载优先顺序(JSP文件)

三:临时修改用户默认语言环境 

国际化的基本步骤

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

struts2:国际化
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=电子邮件地址无效
View Code

以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文件位于java\bin目录下;
  • message_zh_CN_org.properties文件必须是以UTF-8格式保存,这样上述参数“utf-8”才有意义。

最后生成文件如下:

struts2:国际化
title=\u7528\u6237\u6ce8\u518c

name=\u7528\u6237\u540d

username=\u771f\u5b9e\u59d3\u540d

pass=\u5bc6\u7801

repass=\u91cd\u8f93\u5bc6\u7801

sex=\u6027\u522b

province=\u7701

age=\u5e74\u9f84

birth=\u751f\u65e5

love=\u7231\u597d

mobile=\u624b\u673a

email=\u7535\u5b50\u90ae\u4ef6

submit=\u63d0\u4ea4

male=\u7537

famale=\u5973

chongqing=\u91cd\u5e86

beijing=\u5317\u4eac

shanghai=\u4e0a\u6d77

tianjin=\u5929\u6d25

swim=\u6e38\u6cf3

walk=\u6563\u6b65

playtabletennis=\u4e52\u4e53\u7403

reading=\u8bfb\u4e66

others=\u5176\u5b83

validate_name_null=\u8bf7\u8f93\u5165\u7528\u6237\u540d

validate_name_scope=\u7528\u6237\u540d\u5fc5\u987b\u57286\u523018\u4f4d\u4e4b\u95f4

validate_pass_null=\u8bf7\u8f93\u5165\u5bc6\u7801

validate_pass_scope=\u5bc6\u7801\u5fc5\u987b\u57286\u523012\u4f4d\u4e4b\u95f4\uff0c\u4e14\u53ea\u80fd\u662f\u5b57\u6bcd\u6216\u6570\u5b57

validate_repass_null=\u8bf7\u8f93\u5165\u91cd\u590d\u5bc6\u7801

validate_repass_scope=\u4e24\u6b21\u8f93\u5165\u7684\u5bc6\u7801\u5fc5\u987b\u4e00\u81f4

validate_age_scope=\u5e74\u9f84\u5fc5\u987b\u5728 ${min} \u548c ${max}\u4e4b\u95f4

validate_birth_scope=\u751f\u65e5\u5fc5\u987b\u5728 ${min} \u548c ${max}\u4e4b\u95f4

validate_email_scope=\u7535\u5b50\u90ae\u4ef6\u5730\u5740\u65e0\u6548
View Code

2.3 生成英文message_en_US.properties

struts2:国际化
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
View Code

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)

struts2:国际化
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;

    }

}
View Code

其实这个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

结果如下:

struts2:国际化 

直接提交,结果如下:

struts2:国际化 

改变本机语言为英文,测试结果为:

struts2:国际化

注意:修改本机语言的方法,参考附录:二:临时修改用户默认语言环境 

附录

一:国际化资源文件加载优先顺序(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(代表国家、语言环境)。

你可能感兴趣的:(struts2)