在Struts 2.0中國際化(i18n)您的應(yīng)用程序 (ZT)
國際化是商業(yè)系統(tǒng)中不可或缺的一部分,所以無論您學(xué)習(xí)的是什么Web框架,它都是必須掌握的技能。
其實(shí),Struts 1.x在此部分已經(jīng)做得相當(dāng)不錯(cuò)了。它極大地簡化了我們程序員在做國際化時(shí)所需的工作,例如,如果您要輸出一條國際化的信息,只需在代碼包中加入FILE-NAME_xx_XX.properties(其中FILE-NAME為默認(rèn)資源文件的文件名),然后在struts-config.xml中指明其路徑,再在頁面用<bean:message>標(biāo)志輸出即可。
不過,所謂“沒有最好,只有更好”。Struts 2.0并沒有在這部分止步,而是在原有的簡單易用的基礎(chǔ)上,將其做得更靈活、更強(qiáng)大。
國際化Hello World
下面讓我們看一個(gè)例子——HelloWorld。這個(gè)例子演示如何根據(jù)用戶瀏覽器的設(shè)置輸出相應(yīng)的HelloWorld。
- 在Eclipse創(chuàng)建工程配置開發(fā)和運(yùn)行環(huán)境(如果對(duì)這個(gè)步驟有問題,可以參考我早前的文章《為Struts 2.0做好準(zhǔn)備》)。
- 在src文件夾中加入struts.properties文件,內(nèi)容如下:
struts.custom.i18n.resources=globalMessagesStruts 2.0有兩個(gè)配置文件,struts.xml和struts.properties都是放在WEB-INF/classes/下。 - struts.xml用于應(yīng)用程序相關(guān)的配置
- struts.properties用于Struts 2.0的運(yùn)行時(shí)(Runtime)的配置
- 在src文件夾中加入globalMessages_en_US.properties文件,內(nèi)容如下:
HelloWorld=Hello World! - 在src文件夾中加入globalMessages_zh_CN.properties文件,內(nèi)容如下:
HelloWorld=你好,世界!在此想和大家分享一個(gè)不錯(cuò)的編寫properties文件的Eclipse插件(plugin),有了它我們?cè)诰庉嬕恍┖嗴w中文、繁體中文等Unicode文本時(shí),就不必再使用native2ascii編碼了。您可以通過Eclipse中的軟件升級(jí)(Software Update)安裝此插件,步驟如下:
1、展開Eclipse的Help菜單,將鼠標(biāo)移到Software Update子項(xiàng),在出現(xiàn)的子菜單中點(diǎn)擊Find and Install;
2、在Install/Update對(duì)話框中選擇Search for new features to install,點(diǎn)擊Next;
3、在Install對(duì)話框中點(diǎn)擊New Remote Site;
4、在New Update Site對(duì)話框的Name填入“PropEdit”或其它任意非空字符串,在URL中填入http://propedit.sourceforge.jp/eclipse/updates/;
5、在Site to include to search列表中,除上一步加入的site外的其它選項(xiàng)去掉,點(diǎn)擊Finsih;
6、在彈出的Updates對(duì)話框中的Select the features to install列表中將所有結(jié)尾為“3.1.x”的選項(xiàng)去掉(適用于Eclipse 3.2版本的朋友);
7、點(diǎn)擊Finish關(guān)閉對(duì)話框;
8、在下載后,同意安裝,再按提示重啟Eclipse,在工具條看到形似vi的按鈕表示安裝成功,插件可用。此時(shí),Eclpise中所有properties文件的文件名前有綠色的P的圖標(biāo)作為標(biāo)識(shí)。 - 在WebContent文件夾下加入HelloWorl.jsp文件,內(nèi)容如下:
<%@ page contentType="text/html; charset=UTF-8"%>
<%@taglib prefix="s" uri="/struts-tags"%>
<html>
<head>
<title>Hello World</title>
</head>
<body>
<h2><s:text name="HelloWorld"/></h2>
<h2><s:property value="%{getText('HelloWorld')}"/></h2>
</body>
</html> - 發(fā)布運(yùn)行應(yīng)用程序,在瀏覽器地址欄中輸入http://localhost:8080/Struts2_i18n/HelloWorld.jsp ,出現(xiàn)圖1所示頁面。
圖1 中文輸出 - 將瀏覽器的默認(rèn)語言改為“英語(美國)”,刷新頁面,出現(xiàn)圖2所示頁面。
圖2 英文輸出
資源文件查找順序
之所以說Struts 2.0的國際化更靈活是因?yàn)樗梢阅芨鶕?jù)不同需要配置和獲取資源(properties)文件。在Struts 2.0中有下面幾種方法:
- 使用全局的資源文件,方法如上例所示。這適用于遍布于整個(gè)應(yīng)用程序的國際化字符串,它們?cè)诓煌陌╬ackage)中被引用,如一些比較共用的出錯(cuò)提示;
- 使用包范圍內(nèi)的資源文件。做法是在包的根目錄下新建名的package.properties和package_xx_XX.properties文件。這就適用于在包中不同類訪問的資源;
- 使用Action范圍的資源文件。做法為Action的包下新建文件名(除文件擴(kuò)展名外)與Action類名同樣的資源文件。它只能在該Action中訪問。如此一來,我們就可以在不同的Action里使用相同的properties名表示不同的值。例如,在ActonOne中title為“動(dòng)作一”,而同樣用title在ActionTwo表示“動(dòng)作二”,節(jié)省一些命名工夫;
- 使用<s:i18n>標(biāo)志訪問特定路徑的properties文件。使用方法請(qǐng)參考我早前的文章《常用的Struts 2.0的標(biāo)志(Tag)介紹》。在您使用這一方法時(shí),請(qǐng)注意<s:i18n>標(biāo)志的范圍。在<s:i18n name="xxxxx">到</s:i18n>之間,所有的國際化字符串都會(huì)在名為xxxxx資源文件查找,如果找不到,Struts 2.0就會(huì)輸出默認(rèn)值(國際化字符串的名字)。
上面我列舉了四種配置和訪問資源的方法,它們的范圍分別是從大到小,而Struts 2.0在查找國際化字符串所遵循的是特定的順序,如圖3所示:

圖3 資源文件查找順序圖
假設(shè)我們?cè)谀硞€(gè)ChildAction中調(diào)用了getText("user.title"),Struts 2.0的將會(huì)執(zhí)行以下的操作:
- 查找ChildAction_xx_XX.properties文件或ChildAction.properties;
- 查找ChildAction實(shí)現(xiàn)的接口,查找與接口同名的資源文件MyInterface.properties;
- 查找ChildAction的父類ParentAction的properties文件,文件名為ParentAction.properties;
- 判斷當(dāng)前ChildAction是否實(shí)現(xiàn)接口ModelDriven。如果是,調(diào)用getModel()獲得對(duì)象,查找與其同名的資源文件;
- 查找當(dāng)前包下的package.properties文件;
- 查找當(dāng)前包的父包,直到最頂層包;
- 在值棧(Value Stack)中,查找名為user的屬性,轉(zhuǎn)到user類型同名的資源文件,查找鍵為title的資源;
- 查找在struts.properties配置的默認(rèn)的資源文件,參考例1;
- 輸出user.title。
參數(shù)化國際化字符串
許多情況下,我們都需要在動(dòng)行時(shí)(runtime)為國際化字符插入一些參數(shù),例如在輸入驗(yàn)證提示信息的時(shí)候。在Struts 2.0中,我們通過以下兩種方法做到這點(diǎn):
- 在資源文件的國際化字符串中使用OGNL,格式為${表達(dá)式},例如:
validation.require=${getText(fileName)} is required - 使用java.text.MessageFormat中的字符串格式,格式為{ 參數(shù)序號(hào)(從0開始), 格式類形(number | date | time | choice), 格式樣式},例如:
validation.between=Date must between {0, date, short} and {1, date, short}
- 使用標(biāo)志的value0、value1...valueN的屬性,如:
<s:text name="validation.required" value0="User Name"/> - 使用param子元素,這些param將按先后順序,代入到國際化字符串的參數(shù)中,例如:
<s:text name="validation.required">
<s:param value="User Name"/>
</s:text>
讓用戶方便地選擇語言
開發(fā)國際化的應(yīng)用程序時(shí),有一個(gè)功能是必不可少的——讓用戶快捷地選擇或切換語言。在Struts 2.0中,通過ActionContext.getContext().setLocale(Locale arg)可以設(shè)置用戶的默認(rèn)語言。不過,由于這是一個(gè)比較普遍的應(yīng)用場景(Scenario),所以Struts 2.0為您提供了一個(gè)名i18n的攔截器(Interceptor),并在默認(rèn)情況下將其注冊(cè)到攔截器鏈(Interceptor chain)中。它的原理為在執(zhí)行Action方法前,i18n攔截器查找請(qǐng)求中的一個(gè)名為"request_locale"的參數(shù)。如果其存在,攔截器就將其作為參數(shù)實(shí)例化Locale對(duì)象,并將其設(shè)為用戶默認(rèn)的區(qū)域(Locale),最后,將此Locale對(duì)象保存在session的名為“WW_TRANS_I18N_LOCALE”的屬性中。
下面,我將提供一完整示例演示它的使用方法。


















<script type="text/javascript">
<!--
function langSelecter_onChanged() {
document.langForm.submit();
}
//-->
</script>
<s:set name="SESSION_LOCALE" value="#session['WW_TRANS_I18N_LOCALE']"/>
<s:bean id="locales" name="tutorial.Locales"/>
<form action="<s:url includeParams="get" encode="true"/>" name="langForm"
style="background-color: powderblue; padding-top: 4px; padding-bottom: 4px;">
Language: <s:select label="Language"
list="#locales.locales" listKey="value" listValue="key"
value="#SESSION_LOCALE == null ? locale : #SESSION_LOCALE"
name="request_locale" id="langSelecter"
onchange="langSelecter_onChanged()" theme="simple"/>
</form>
上述代碼的原理為,LangSelector.jsp先實(shí)例化一個(gè)Locales對(duì)象,并把對(duì)象的Map類型的屬性locales賦予下拉列表(select) 。如此一來,下拉列表就獲得可用語言的列表。大家看到LangSelector有<s:form>標(biāo)志和一段Javascript腳本,它們的作用就是在用戶在下拉列表中選擇了后,提交包含“reqeust_locale”變量的表單到Action。在打開頁面時(shí),為了下拉列表的選中的當(dāng)前區(qū)域,我們需要到session取得當(dāng)前區(qū)域(鍵為“WW_TRANS_I18N_LOCALE”的屬性),而該屬性在沒有設(shè)置語言前是為空的,所以通過值棧中l(wèi)ocale屬性來取得當(dāng)前區(qū)域(用戶瀏覽器所設(shè)置的語言)。
你可以把LangSelector.jsp作為一個(gè)控件使用,方法是在JSP頁面中把它包含進(jìn)來,代碼如下所示:在例1中的HellloWorld.jsp中<body>后加入上述代碼,并在struts.xml中新建Action,代碼如下:
<result>/HelloWorld.jsp</result>
</action>
或者,如果你多個(gè)JSP需要實(shí)現(xiàn)上述功能,你可以使用下面的通用配置,而不是為每一個(gè)JSP頁面都新建一個(gè)Action。
<result>/{1}.jsp</result>
</action>
分布運(yùn)行程序,在瀏覽器的地址欄中輸入http://localhost:8080/Struts2_i18n/HelloWorld.action,出現(xiàn)圖4所示頁面:

圖3 HelloWorld.action
在下拉列表中,選擇“American English”,出現(xiàn)圖5所示頁面:

圖4 HelloWorld.action
![]() |
可能大家會(huì)問為什么一定要通過Action來訪問頁面呢? 你可以試一下不用Action而直接用JSP的地址來訪問頁面,結(jié)果會(huì)是無論你在下拉列表中選擇什么,語言都不會(huì)改變。這表示不能正常運(yùn)行的。其原因?yàn)槿绻苯邮褂肑SP訪問頁面,Struts 2.0在web.xml的配置的過濾器(Filter)就不會(huì)工作,所以攔截器鏈也不會(huì)工作。 |
posted on 2007-05-16 17:13 風(fēng)人園 閱讀(223) 評(píng)論(0) 編輯 收藏 所屬分類: Struts