基于請求URL的國際化實現(xiàn)方式

基于請求URL的國際化實現(xiàn)方式

原理:使用spring的request bean保存相應(yīng)的國際化組件,這樣保證同一個請求的國際化相同继阻,也是在微服務(wù)處理國際化的一種方式示括。需要針對每個請求做不同的國際化

實現(xiàn)所以需要姆怪,相應(yīng)的攔截器去處理對應(yīng)請求域中的國際化組件

  1. 配置文件:
  • spring容器配置applicationContext.xml溯乒,中添加取得信息的messageSource,放在spring容器而非springmvc容器加載是因為代碼中有Service的注解依賴于他
<!-- 國際換的service依賴于他匪煌,所以從mvc提到前面 -->
    <bean id="messageSource"
        class="org.springframework.context.support.ResourceBundleMessageSource">
        <!-- 國際化信息所在的文件名 -->
        <property name="basename" value="messages/messages" />
        <property name="defaultEncoding" value="UTF-8"/> 
        <!-- 如果在國際化資源文件中找不到對應(yīng)代碼的信息责蝠,就用這個代碼作為名稱 -->
        <property name="useCodeAsDefaultMessage" value="true" />
    </bean>
  1. springmvc容器配置攔截器,注意攔截器的順序萎庭,國際化的攔截器在處理請求的攔截器前面
<mvc:interceptors>
       <mvc:interceptor>  
      <mvc:mapping path="/**"/>  
       <!-- 國際化操作攔截器 如果采用基于(請求/Session/Cookie)則必需配置 --> 
      <bean  id="localeChangeInterceptor"
               class="org.springframework.web.servlet.i18n.LocaleChangeInterceptor">
          <property name="paramName" value="locale"/>
      </bean>
      </mvc:interceptor>  
      <mvc:interceptor>
          <!-- 需攔截的地址 -->
          <!--   級目錄 -->
          <mvc:mapping path="/*" />
          <mvc:mapping path="/*/*" />
          <!-- 需排除攔截的地址 -->
          <mvc:exclude-mapping path="/*.html"/>
          <bean class="cn.xx.xx.xx.interceptor.ControllerInterceptor" />
      </mvc:interceptor>
  </mvc:interceptors>
  <!-- 基于url的國際化 id必須為localeResolver否則國際化組件無法識別霜医,UrlAcceptHeaderLocaleResolver為自定義實現(xiàn)部分-->
<bean id="localeResolver" class="cn.abcsys.devops.application.service.UrlAcceptHeaderLocaleResolver"/>
  1. UrlAcceptHeaderLocaleResolver作為localeResolver國際urlLocal
/**
* Copyright: Copyright (c) 2018 LanRu-Caifu
* @author xzg
* 2018年2月6日
* @ClassName: UrlAcceptHeaderLocaleResolver.java
* @Description: 國際化攔截請求后對請求更改Local
* @version: v1.0.0
*/
public class UrlAcceptHeaderLocaleResolver extends AcceptHeaderLocaleResolver {

   private Locale urlLocal;

   public Locale resolveLocale(HttpServletRequest request) {
       
       return urlLocal != null?urlLocal:request.getLocale();
   } 
   @Override
   public void setLocale(HttpServletRequest request, HttpServletResponse response, Locale locale) {
       urlLocal = locale;
   }
}
  1. spring中的request bean依賴于接口實現(xiàn),下面是其接口和對應(yīng)的實現(xiàn)類
public interface I18nSessionService {
  
  public void setRc(RequestContext rc);
  
  public void setRcByRequest(HttpServletRequest request);
  
  public String getMessage(String key); 
  
  public String getMessage(String key,Object info);
  
  public String getMessage(String key,Object...objects);
}
  • 下面注解主要為設(shè)置作用域為request擎椰,注入messageSource組件支子,并提供RequestContext用于切換語言配置國際化
@Component
@RequestScope(proxyMode = ScopedProxyMode.INTERFACES)
public class I18nSessionServiceImpl implements I18nSessionService {

  @Autowired  
  @Qualifier("messageSource")  
  private MessageSource resources;
  //前端設(shè)置切換語言是設(shè)置
  private RequestContext rc ;
  
  public RequestContext getRc() {
      return rc;
  }
  @Override
  public void setRcByRequest(HttpServletRequest request) {
      // TODO Auto-generated method stub
      this.rc = new RequestContext(request);
  }
  
  @Override
  public void setRc(RequestContext rc) {
      this.rc = rc;
  }

  @Override
  public String getMessage(String key) {
      // TODO Auto-generated method stub
      if(null != getRc()){
          return getRc().getMessage(key);
      }
      return resources.getMessage(key, null, null);
  }

  @Override
  public String getMessage(String key, Object info) {
      // TODO Auto-generated method stub
      if(null != getRc()){
          return getRc().getMessage(key, new Object[]{info});
      }
      return resources.getMessage(key, new Object[]{info}, null);
  }

  @Override
  public String getMessage(String key, Object... objects) {
              if(null != getRc()){
                  return getRc().getMessage(key, objects);
              }
              return resources.getMessage(key, objects, null);
  }
}
  1. 自定義的攔截器中處理创肥,國際化組件的請求bean
@Component
public class ControllerInterceptor implements HandlerInterceptor {
  /**
   * 在Controller方法前進(jìn)行攔截
   */
  @Resource
  private I18nSessionService  i18nSessionService;
  private static Logger log = Logger.getLogger("ControllerInterceptor");
  public boolean preHandle(HttpServletRequest request,
                           HttpServletResponse response, Object handler) throws Exception {
      log.info("RequestURI :"+request.getRequestURI());
      //解決跨域
      response.setHeader("Access-Control-Allow-Origin", "*");
      response.setHeader("Access-Control-Allow-Methods","POST");
      response.setHeader("Access-Control-Allow-Headers","x-requested-with,content-type");
      //攔截器中對所有的請求處理达舒,保存到request bean中
      i18nSessionService.setRcByRequest(request);
      return true;
  }

  public void postHandle(HttpServletRequest request,
                         HttpServletResponse response, Object handler,
                         ModelAndView modelAndView) throws Exception {
  }
  /**
   * 在Controller方法后進(jìn)行攔截
   */
  public void afterCompletion(HttpServletRequest request,
                              HttpServletResponse response, Object handler, Exception ex)
          throws Exception {
  }
}
  1. 使用方式
//發(fā)送請求 http://localhost:8080/testI18n.do?locale=en_US 或者h(yuǎn)ttp://localhost:8080/testI18n.do?locale=zh_CN
@Resource
private I18nSessionService is;
@RequestMapping(value = "/testI18n.do", method = { RequestMethod.POST,RequestMethod.GET})
public @ResponseBody
   Result testI18n(){
       return new Result(true, is.getMessage("argument.required"), "");
}
  • 總結(jié):以上就是基本實現(xiàn)過程。在微服務(wù)中由于服務(wù)發(fā)現(xiàn)提供的服務(wù)模塊會自適應(yīng)調(diào)整所以不適合使用session 的方式處理國際化叹侄。這里使用request和url將粒度劃分的更細(xì)巩搏,處理也更靈活
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市趾代,隨后出現(xiàn)的幾起案子贯底,更是在濱河造成了極大的恐慌,老刑警劉巖撒强,帶你破解...
    沈念sama閱讀 221,198評論 6 514
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件禽捆,死亡現(xiàn)場離奇詭異,居然都是意外死亡飘哨,警方通過查閱死者的電腦和手機(jī)胚想,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 94,334評論 3 398
  • 文/潘曉璐 我一進(jìn)店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來芽隆,“玉大人浊服,你說我怎么就攤上這事统屈。” “怎么了牙躺?”我有些...
    開封第一講書人閱讀 167,643評論 0 360
  • 文/不壞的土叔 我叫張陵愁憔,是天一觀的道長。 經(jīng)常有香客問我孽拷,道長吨掌,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 59,495評論 1 296
  • 正文 為了忘掉前任脓恕,我火速辦了婚禮思犁,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘进肯。我一直安慰自己激蹲,他們只是感情好,可當(dāng)我...
    茶點故事閱讀 68,502評論 6 397
  • 文/花漫 我一把揭開白布江掩。 她就那樣靜靜地躺著学辱,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 52,156評論 1 308
  • 那天淤毛,我揣著相機(jī)與錄音田盈,去河邊找鬼。 笑死谍憔,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的。 我是一名探鬼主播危队,決...
    沈念sama閱讀 40,743評論 3 421
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼钙畔!你這毒婦竟也來了茫陆?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,659評論 0 276
  • 序言:老撾萬榮一對情侶失蹤擎析,失蹤者是張志新(化名)和其女友劉穎簿盅,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體揍魂,經(jīng)...
    沈念sama閱讀 46,200評論 1 319
  • 正文 獨居荒郊野嶺守林人離奇死亡桨醋,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 38,282評論 3 340
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了现斋。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片喜最。...
    茶點故事閱讀 40,424評論 1 352
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖步责,靈堂內(nèi)的尸體忽然破棺而出返顺,到底是詐尸還是另有隱情禀苦,我是刑警寧澤,帶...
    沈念sama閱讀 36,107評論 5 349
  • 正文 年R本政府宣布遂鹊,位于F島的核電站振乏,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏秉扑。R本人自食惡果不足惜慧邮,卻給世界環(huán)境...
    茶點故事閱讀 41,789評論 3 333
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望舟陆。 院中可真熱鬧误澳,春花似錦、人聲如沸秦躯。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,264評論 0 23
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽踱承。三九已至倡缠,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間茎活,已是汗流浹背昙沦。 一陣腳步聲響...
    開封第一講書人閱讀 33,390評論 1 271
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點兒被人妖公主榨干…… 1. 我叫王不留载荔,地道東北人盾饮。 一個月前我還...
    沈念sama閱讀 48,798評論 3 376
  • 正文 我出身青樓,卻偏偏與公主長得像懒熙,于是被迫代替她去往敵國和親丘损。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,435評論 2 359

推薦閱讀更多精彩內(nèi)容