網(wǎng)站開發(fā)流程書籍上海百網(wǎng)優(yōu)seo優(yōu)化公司
1. 簡(jiǎn)介
Spring國(guó)際化(Spring Internationalization,簡(jiǎn)稱i18n)是Spring框架提供的一種機(jī)制,用于支持多語言的應(yīng)用程序。它使得開發(fā)者能夠輕松地在應(yīng)用程序中實(shí)現(xiàn)不同語言的支持,從而滿足全球化的需求。通過Spring國(guó)際化,開發(fā)者可以將應(yīng)用程序的文本、標(biāo)簽、消息等資源抽取出來,并使用合適的語言文件進(jìn)行翻譯,使得應(yīng)用程序能夠根據(jù)用戶的語言偏好自動(dòng)切換語言。這種機(jī)制不僅簡(jiǎn)化了多語言支持的實(shí)現(xiàn),還使得應(yīng)用程序更加易于維護(hù)和擴(kuò)展。在Spring國(guó)際化的實(shí)現(xiàn)中,主要涉及到了MessageSource、LocaleResolver等核心組件,它們共同協(xié)作,實(shí)現(xiàn)了語言切換的功能。通過使用Spring國(guó)際化的API,開發(fā)者可以方便地定義語言區(qū)域、加載資源文件、處理消息等操作,從而快速構(gòu)建多語言的應(yīng)用程序。
2. API介紹
ApplicationContext 接口擴(kuò)展了一個(gè)名為 MessageSource 的接口,因此提供了國(guó)際化("i18n")功能。Spring 還提供了 HierarchicalMessageSource 接口,該接口可以分層解析消息。這些接口共同構(gòu)成了 Spring 實(shí)現(xiàn)消息解析的基礎(chǔ)。這些接口定義的方法包括:
-
String getMessage(String code, Object[] args, String default, Locale loc)
用于從 MessageSource 獲取消息的基本方法。如果在指定的本地沒有找到消息,則使用默認(rèn)消息。通過標(biāo)準(zhǔn)庫(kù)提供的 MessageFormat 功能,傳入的任何參數(shù)都會(huì)成為替換值。
-
String getMessage(String code, Object[] args, Locale loc)
與前一種方法基本相同,但有一點(diǎn)不同:不能指定默認(rèn)信息。如果找不到信息,就會(huì)拋出 NoSuchMessageException 異常。
-
String getMessage(MessageSourceResolvable resolvable, Locale locale)
前面方法中使用的所有屬性也都封裝在一個(gè)名為 MessageSourceResolvable 的類中,你可以使用該方法。
3. 國(guó)際化初始化
Spring容器ApplicationContext初始化過程中,會(huì)從容器中查找MessageSource類型的Bean。并且該Bean的名稱必須是 messageSource。如果找到了這樣一個(gè) Bean,對(duì)前面方法的所有調(diào)用都會(huì)委托給消息源。如果沒有找到消息源,ApplicationContext 會(huì)嘗試查找包含同名Bean的父類。如果找到了,它就會(huì)使用該 bean 作為消息源。如果 ApplicationContext 無法找到任何消息源,則會(huì)實(shí)例化一個(gè)空的 DelegatingMessageSource,以便能夠接受對(duì)上述方法的調(diào)用。
public abstract class AbstractApplicationContext {public void refresh() {// 初始化消息源initMessageSource();}/*** 初始化消息源。* 如果當(dāng)前上下文中沒有定義消息源,則使用父級(jí)消息源。*/
protected void initMessageSource() {ConfigurableListableBeanFactory beanFactory = getBeanFactory();if (beanFactory.containsLocalBean(MESSAGE_SOURCE_BEAN_NAME)) {this.messageSource = beanFactory.getBean(MESSAGE_SOURCE_BEAN_NAME, MessageSource.class);// 使消息源知道父級(jí)消息源。if (this.parent != null && this.messageSource instanceof HierarchicalMessageSource hms &&hms.getParentMessageSource() == null) {// 只有當(dāng)父級(jí)消息源尚未注冊(cè)時(shí),才將父上下文設(shè)置為父級(jí)消息源。hms.setParentMessageSource(getInternalParentMessageSource());}if (logger.isTraceEnabled()) {logger.trace("使用的消息源為 [" + this.messageSource + "]");}}else {// 使用空消息源以能夠接受getMessage調(diào)用。DelegatingMessageSource dms = new DelegatingMessageSource();dms.setParentMessageSource(getInternalParentMessageSource());this.messageSource = dms;beanFactory.registerSingleton(MESSAGE_SOURCE_BEAN_NAME, this.messageSource);if (logger.isTraceEnabled()) {logger.trace("沒有'" + MESSAGE_SOURCE_BEAN_NAME + "' bean,使用 [" + this.messageSource + "]");}}
}
}
4. 國(guó)際化配置
基于Spring環(huán)境
@Bean(AbstractApplicationContext.MESSAGE_SOURCE_BEAN_NAME)
public MessageSource messageSource() {ReloadableResourceBundleMessageSource messageSource = new ReloadableResourceBundleMessageSource() ;// 這里設(shè)置的是basename,message是文件的前綴(不是包)messageSource.addBasenames("classpath:com/pack/main/databinder/message") ;return messageSource ;
}
在包c(diǎn)om/pack/main/databinder下建2個(gè)文件分別:message_zh_CN.properties和message_en_US.properties。文件內(nèi)容如下:
message_zh_CN.properties
#姓名必須填寫
user.name.empty=\u59D3\u540D\u5FC5\u987B\u586B\u5199
message_en_US.properties
user.name.empty=name is required
調(diào)用
try (AnnotationConfigApplicationContext context = new AnnotationConfigApplicationContext(AppConfig.class)) {// Locale.CHINA或者Locale.USSystem.out.println(context.getMessage("user.name.empty", null, Locale.CHINA)) ;
}
基于SpringBoot環(huán)境
spring:messages:basename: message
注意:你需要提供一個(gè)默認(rèn)的message.properties文件
@RestController
@RequestMapping("/i18n")
public class I18NController {@Resourceprivate ApplicationContext context ;@GetMapping("/index")public String index() {return context.getMessage("user.name.empty", null, "默認(rèn)消息", LocaleContextHolder.getLocale()) ;}}
Locale從當(dāng)前線程上下文中獲取。該Locale是在DispatcherServlet中初始化的。
在接口調(diào)用時(shí),我們只需要指定Access-Language header
5. 其它配置
Spring為我們提供了一個(gè)便捷的類,可以更方便的訪問消息源,項(xiàng)目中只需要注冊(cè)如下bean:
@Bean
public MessageSourceAccessor messageSourceAccessor(MessageSource messageSource) {MessageSourceAccessor accessor = new MessageSourceAccessor(messageSource) ;return accessor ;
}
訪問
@Resource
private MessageSourceAccessor accessor ;
@GetMapping("/index")
public String index() {return accessor.getMessage("user.name.empty") ;
}
帶占位符的消息訪問
在消息文件中定義如下:
#年齡的取值范圍從{0}~{1}
user.age.range=\u5E74\u9F84\u7684\u53D6\u503C\u8303\u56F4\u4ECE{0}~{1}
訪問
@GetMapping("/index")
public String index() {return accessor.getMessage("user.age.range", new Object[] {1, 100}) ;
}
注:Spring 還提供了一個(gè)ReloadableResourceBundleMessageSource?類。該變體支持相同的捆綁文件格式,但比基于 JDK 的標(biāo)準(zhǔn) ResourceBundleMessageSource 實(shí)現(xiàn)更靈活。特別是,它允許從任何 Spring 資源位置(而不僅僅是從類路徑)讀取文件,并支持捆綁屬性文件的熱重載(同時(shí)在兩者之間有效地緩存它們)。