海鷗航際

          JAVA站
          posts - 11, comments - 53, trackbacks - 1, articles - 102

          Struts 原理與實踐(5)

          Posted on 2005-03-23 17:57 海天一鷗 閱讀(377) 評論(0)  編輯  收藏 所屬分類: Struts專題
          一個支持i18n的應用程序應該有如下一些特征:
          1增加支持的語言時要求不更改程序代碼
          2字符元素、消息、和圖象保存在原代碼之外
          3依賴于不同文化的數據如:日期時間、小數、及現金符號等數據對用戶的語言和地理位置應該有正確的格式
          4應用程序能迅速地適應新語言和/或新地區

          Struts主要采用兩個i18n組件來實現國際化編程:

          第一個組件是一個被應用程序控制器管理的消息類,它引用包含地區相關信息串的資源包。第二個組件是一個JSP定制標簽,,它用于在View層呈現被控制器管理的實際的字符串。在我們前面的登錄例子中這兩方面的內容都出現過。

          用Struts實現國際化編程的標準做法是:生成一個java屬性文件集。每個文件包含您的應用程序要顯示的所有消息的鍵/值對。

          這些文件的命名要遵守如下規則,代表英文消息的文件可作為缺省的文件,它的名稱是ApplicationResources.properties;其他語種的文件在文件名中都要帶上相應的地區和語言編碼串,如代表中文的文件名應為ApplicationResources_zh_CN.properties。并且其他語種的文件與ApplicationResources.properties文件要放在同一目錄中。

          ApplicationResources.properties文件的鍵/值都是英文的,而其他語種文件的鍵是英文的,值則是對應的語言。如在我們前面的登錄例子中的鍵/值對:logon.jsp.prompt.username=Username:在中文文件中就是:logon.jsp.prompt.username=用戶名:當然,在實際應用時要把中文轉換為AscII碼。

          有了上一篇文章和以上介紹的一些基礎知識后。我們就可以將我們的登錄程序進行國際化編程了。

          首先,我們所有jsp頁面文件的字符集都設置為UTF-8。即在頁面文件的開始寫如下指令行:

          <%@ page contentType="text/html; charset=UTF-8" %>,在我們的登錄例子中已經這樣做了,這里不需要再改動。

          其次,將所有的request的字符集也設置為UTF-8。雖然,我們可以在每個文件中加入這樣的句子:request.setCharacterEncoding("UTF-8");來解決,但這樣顯得很麻煩。一種更簡單的解決方法是使用filter。具體步驟如下:

          在mystruts\WEB-INF\classes目錄下再新建一個名為filters的目錄,新建一個名為:SetCharacterEncodingFilter的類,并保存在該目錄下。其實,這個類并不要您親自來寫,可以借用tomcat中的例子。現將該例子的程序節選如下:

          package filters;
          import java.io.IOException;
          import javax.servlet.Filter;
          import javax.servlet.FilterChain;
          import javax.servlet.FilterConfig;
          import javax.servlet.ServletException;
          import javax.servlet.ServletRequest;
          import javax.servlet.ServletResponse;
          import javax.servlet.UnavailableException;
          
          /**
           * <p>Example filter that sets the character encoding to be used in parsing the
           * incoming request, either unconditionally or only if the client did not
           * specify a character encoding.  Configuration of this filter is based on
           * the following initialization parameters:</p>
           * <ul>
           * <li><strong>encoding</strong> - The character encoding to be configured
           *     for this request, either conditionally or unconditionally based on
           *     the <code>ignore</code> initialization parameter.  This parameter
           *     is required, so there is no default.</li>
           * <li><strong>ignore</strong> - If set to "true", any character encoding
           *     specified by the client is ignored, and the value returned by the
           *     <code>selectEncoding()</code> method is set.  If set to "false,
           *     <code>selectEncoding()</code> is called <strong>only</strong> if the
           *     client has not already specified an encoding.  By default, this
           *     parameter is set to "true".</li>
           * </ul>
           *
           * <p>Although this filter can be used unchanged, it is also easy to
           * subclass it and make the <code>selectEncoding()</code> method more
           * intelligent about what encoding to choose, based on characteristics of
           * the incoming request (such as the values of the <code>Accept-Language</code>
           * and <code>User-Agent</code> headers, or a value stashed in the current
           * user's session.</p>
           *
           * @author Craig McClanahan
           * @version $Revision: 1.2 $ $Date: 2001/10/17 22:53:19 $
           */
          
          public class SetCharacterEncodingFilter implements Filter {
          
          
              // ----------------------------------------------------- Instance Variables
          
          
              /**
               * The default character encoding to set for requests that pass through
               * this filter.
               */
              protected String encoding = null;
          
          
              /**
               * The filter configuration object we are associated with.  If this value
               * is null, this filter instance is not currently configured.
               */
              protected FilterConfig filterConfig = null;
          
          
              /**
               * Should a character encoding specified by the client be ignored?
               */
              protected boolean ignore = true;
          
          
              // --------------------------------------------------------- Public Methods
          
          
              /**
               * Take this filter out of service.
               */
              public void destroy() {
          
                  this.encoding = null;
                  this.filterConfig = null;
          
              }
          
          
              /**
               * Select and set (if specified) the character encoding to be used to
               * interpret request parameters for this request.
               *
               * @param request The servlet request we are processing
               * @param result The servlet response we are creating
               * @param chain The filter chain we are processing
               *
               * @exception IOException if an input/output error occurs
               * @exception ServletException if a servlet error occurs
               */
              public void doFilter(ServletRequest request, ServletResponse response,
                                   FilterChain chain)
                  throws IOException, ServletException {
          
                  // Conditionally select and set the character encoding to be used
                  if (ignore || (request.getCharacterEncoding() == null)) {
                      String encoding = selectEncoding(request);
                      if (encoding != null)
                          request.setCharacterEncoding(encoding);
                  }
          
                  // Pass control on to the next filter
                  chain.doFilter(request, response);
          
              }
          
          
              /**
               * Place this filter into service.
               *
               * @param filterConfig The filter configuration object
               */
              public void init(FilterConfig filterConfig) throws ServletException {
          
                  this.filterConfig = filterConfig;
                  this.encoding = filterConfig.getInitParameter("encoding");
                  String value = filterConfig.getInitParameter("ignore");
                  if (value == null)
                      this.ignore = true;
                  else if (value.equalsIgnoreCase("true"))
                      this.ignore = true;
                  else if (value.equalsIgnoreCase("yes"))
                      this.ignore = true;
                  else
                      this.ignore = false;
          
              }
          
          
              // ------------------------------------------------------ Protected Methods
          
          
              /**
               * Select an appropriate character encoding to be used, based on the
               * characteristics of the current request and/or filter initialization
               * parameters.  If no character encoding should be set, return
               * <code>null</code>.
               * <p>
               * The default implementation unconditionally returns the value configured
               * by the <strong>encoding</strong> initialization parameter for this
               * filter.
               *
               * @param request The servlet request we are processing
               */
              protected String selectEncoding(ServletRequest request) {
          
                  return (this.encoding);
          
              }
          
          }


          其中,request.setCharacterEncoding(encoding);是一個關鍵句子。

          為了讓該類工作,我們還要在web.xml文件中對它進行配置,配置代碼如下:

          <filter>
              <filter-name>Set Character Encoding</filter-name>
              <filter-class>filters.SetCharacterEncodingFilter</filter-class>
              <init-param>
                <param-name>encoding</param-name>
                <param-value>UTF-8</param-value>
              </init-param>
            </filter>
            <filter-mapping>
              <filter-name>Set Character Encoding</filter-name>
              <url-pattern>/*</url-pattern>
            </filter-mapping>


          最后,就是準備資源包文件,我們以創建一個中文文件為例:

          將ApplicationResources.properties文件打開,另存為ApplicationResources_zh.properties,這只是一個過渡性質的文件。將文件中鍵/值對的值都用中文表示。更改完后的代碼如下:

          #Application Resource for the logon.jsp
          logon.jsp.title=登錄頁
          logon.jsp.page.heading=歡迎 世界!
          logon.jsp.prompt.username=用戶名:
          logon.jsp.prompt.password=口令:
          logon.jsp.prompt.submit=提交
          logon.jsp.prompt.reset=復位
          
          #Application Resource for the main.jsp
          main.jsp.title=主頁
          main.jsp.welcome=歡迎:
          
          #Application Resource for the LogonAction.java
          error.missing.username=<li><font color="red">沒有輸入用戶名</font></li>
          error.missing.password=<li><font color="red">沒有輸入口令</font></li>
          
          #Application Resource for the UserInfoBo.java
          error.noMatch=<li><font color="red">沒有匹配的用戶</font></li>
          
          #Application Resource for the UserInfoBo.java
          error.logon.invalid=<li><font color="red">用戶名/口令是無效的</font></li>
          error.removed.user=<li><font color="red">找不到該用戶</font></li>
          error.unexpected=<li><font color="red">不可預期的錯誤</font></li>


          使用native2ascii工具將上面文件中的中文字符轉換為ascii碼,并生成一個最終使用的資源文件ApplicationResources_zh_CN.properties。

          具體做法是打開一個dos窗口,到mystruts\WEB-INF\classes目錄下,運行如下語句:

          native2ascii -encoding GBK ApplicationResources_zh.properties ApplicationResources_zh_CN.properties

          生成的文件ApplicationResources_zh_CN.properties的內容如下:

          #Application Resource for the logon.jsp
          logon.jsp.title=\u767b\u5f55\u9875
          logon.jsp.page.heading=\u6b22\u8fce \u4e16\u754c!
          logon.jsp.prompt.username=\u7528\u6237\u540d:
          logon.jsp.prompt.password=\u53e3\u4ee4:
          logon.jsp.prompt.submit=\u63d0\u4ea4
          logon.jsp.prompt.reset=\u590d\u4f4d
          
          #Application Resource for the main.jsp
          main.jsp.title=\u4e3b\u9875
          main.jsp.welcome=\u6b22\u8fce:
          
          #Application Resource for the LogonAction.java
          error.missing.username=<li><font color="red">\u6ca1\u6709\u8f93\u5165\u7528\u6237\u540d</font></li>
          error.missing.password=<li><font color="red">\u6ca1\u6709\u8f93\u5165\u53e3\u4ee4</font></li>
          
          #Application Resource for the UserInfoBo.java
          error.noMatch=<li><font color="red">\u6ca1\u6709\u5339\u914d\u7684\u7528\u6237</font></li>
          
          #Application Resource for the UserInfoBo.java
          error.logon.invalid=<li><font color="red">\u7528\u6237\u540d/\u53e3\u4ee4\u662f\u65e0\u6548\u7684</font></li>
          error.removed.user=<li><font color="red">\u627e\u4e0d\u5230\u8be5\u7528\u6237</font></li>
          error.unexpected=<li><font color="red">\u4e0d\u53ef\u9884\u671f\u7684\u9519\u8bef</font></li>


          從這里可以看出,所有的中文字都轉換成了對應的Unicode碼。

          現在,再運行登錄例子程序,您會發現它已經是顯示的中文了。在瀏覽器的"工具"--"Internet選項"的"語言首選項"對話框中,去掉"中文(中國)"加上英文,再試登錄程序,此時,又會顯示英文。這就是說不同國家(地區)的客戶都可以看到自己語言的內容,這就實現了國際化編程的基本要求。如果還要顯示其他語言,可采用類似處理中文的方法進行,這里就不細講了。

          本文中的例子程序所采用的數據庫仍然是MS SQLServer2000,數據庫字符集為gbk。實驗表明,對簡、繁體中文,英文及日文字符都能支持。

          參考文獻:
          《Programming Jakarta Struts》Chuck Cavaness著
          《Mastering Jakarta Struts》James Goodwill著
          主站蜘蛛池模板: 姜堰市| 武隆县| 九龙县| 鹤壁市| 台安县| 鄂托克前旗| 太湖县| 长汀县| 苏尼特左旗| 根河市| 博湖县| 兴宁市| 玉环县| 沙河市| 大荔县| 海林市| 贵南县| 阿拉善左旗| 鄂温| 如东县| 文化| 和龙市| 佛坪县| 灵寿县| 长春市| 石首市| 泽州县| 栖霞市| 贵港市| 灵山县| 万山特区| 汉中市| 油尖旺区| 固阳县| 鸡泽县| 佛学| 游戏| 攀枝花市| 盐边县| 海门市| 崇礼县|