javascript
自己动手在Spring-Boot上加强国际化功能
前言
公司將項(xiàng)目由Struts2轉(zhuǎn)到Springmvc了,由于公司業(yè)務(wù)是境外服務(wù),所以對(duì)國(guó)際化功能需求很高。Struts2自帶的國(guó)際化功能相對(duì)Springmvc來(lái)說(shuō)更加完善,不過(guò)spring很大的特性就是可定定制化性強(qiáng),所以在公司項(xiàng)目移植的到Springmvc的時(shí)候增加了其國(guó)際化的功能。特此整理記錄并且完善了一下。
本文主要實(shí)現(xiàn)的功能:
注:本文不詳細(xì)介紹怎么配置國(guó)際化,區(qū)域解析器等。
實(shí)現(xiàn)
國(guó)際化項(xiàng)目初始化
先創(chuàng)建一個(gè)基本的Spring-Boot+thymeleaf+國(guó)際化信息(message.properties)項(xiàng)目,如果有需要可以從我的Github下載。
簡(jiǎn)單看一下項(xiàng)目的目錄和文件
其中I18nApplication.java設(shè)置了一個(gè)CookieLocaleResolver,采用cookie來(lái)控制國(guó)際化的語(yǔ)言。還設(shè)置一個(gè)LocaleChangeInterceptor攔截器來(lái)攔截國(guó)際化語(yǔ)言的變化。
public class I18nApplication {public static void main(String[] args) {SpringApplication.run(I18nApplication.class, args);}public LocaleResolver localeResolver() {CookieLocaleResolver slr = new CookieLocaleResolver();slr.setCookieMaxAge(3600);slr.setCookieName("Language");//設(shè)置存儲(chǔ)的Cookie的name為L(zhǎng)anguagereturn slr;}public WebMvcConfigurer webMvcConfigurer() {return new WebMvcConfigurer() {//攔截器public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new LocaleChangeInterceptor()).addPathPatterns("/**");}};} } 復(fù)制代碼我們?cè)倏匆幌耯ello.html中寫了什么:
<html xmlns="http://www.w3.org/1999/xhtml" xmlns:th="http://www.thymeleaf.org"> <head><title>Hello World!</title> </head> <body> <h1 th:text="#{i18n_page}"></h1> <h3 th:text="#{hello}"></h3> </body> </html> 復(fù)制代碼現(xiàn)在啟動(dòng)項(xiàng)目并且訪問(wèn)http://localhost:9090/hello(我在application.properties)中設(shè)置了端口為9090。
由于瀏覽器默認(rèn)的語(yǔ)言是中文,所以他默認(rèn)會(huì)去messages_zh_CN.properties中找,如果沒(méi)有就會(huì)去messages.properties中找國(guó)際化詞。
然后我們?cè)跒g覽器中輸入http://localhost:9090/hello?locale=en_US,語(yǔ)言就會(huì)切到英文。同樣的如果url后參數(shù)設(shè)置為locale=zh_CH,語(yǔ)言就會(huì)切到中文。
從文件夾中直接加載多個(gè)國(guó)際化文件
在我們hello.html頁(yè)面中,只有'i18n_page'和'hello'兩個(gè)國(guó)際化信息,然而在實(shí)際項(xiàng)目中肯定不會(huì)只有幾個(gè)國(guó)際化信息那么少,通常都是成千上百個(gè)的,那我們肯定不能把這么多的國(guó)際化信息都放在messages.properties一個(gè)文件中,通常都是把國(guó)際化信息分類存放在幾個(gè)文件中。但是當(dāng)項(xiàng)目大了以后,這些國(guó)際化文件也會(huì)越來(lái)越多,這時(shí)候在application.properties文件中一個(gè)個(gè)的去配置這個(gè)文件也是不方便的,所以現(xiàn)在我們實(shí)現(xiàn)一個(gè)功能自動(dòng)加載制定目錄下所有的國(guó)際化文件。
繼承ResourceBundleMessageSource
在項(xiàng)目下創(chuàng)建一個(gè)類繼承ResourceBundleMessageSource或者ReloadableResourceBundleMessageSource,起名為MessageResourceExtension。并且注入到bean中起名為messageSource,這里我們繼承ResourceBundleMessageSource。
("messageSource") public class MessageResourceExtension extends ResourceBundleMessageSource { } 復(fù)制代碼注意這里我們的Component名字必須為'messageSource',因?yàn)樵诔跏蓟疉pplicationContext的時(shí)候,會(huì)查找bean名為'messageSource'的bean。這個(gè)過(guò)程在AbstractApplicationContext.java中,我們看一下源代碼
/** * Initialize the MessageSource. * Use parent's if none defined in this context. */ protected void initMessageSource() {ConfigurableListableBeanFactory beanFactory = getBeanFactory();if (beanFactory.containsLocalBean(MESSAGE_SOURCE_BEAN_NAME)) {this.messageSource = beanFactory.getBean(MESSAGE_SOURCE_BEAN_NAME, MessageSource.class);...} } ... 復(fù)制代碼在這個(gè)初始化MessageSource的方法中,beanFactory查找注入名為MESSAGE_SOURCE_BEAN_NAME(messageSource)的bean,如果沒(méi)有找到,就會(huì)在其父類中查找是否有該名的bean。
實(shí)現(xiàn)文件加載
現(xiàn)在我們可以開(kāi)始在剛才創(chuàng)建的MessageResourceExtension
中寫加載文件的方法了。
("messageSource") public class MessageResourceExtension extends ResourceBundleMessageSource {private final static Logger logger = LoggerFactory.getLogger(MessageResourceExtension.class);/*** 指定的國(guó)際化文件目錄*/(value = "${spring.messages.baseFolder:i18n}")private String baseFolder;/*** 父MessageSource指定的國(guó)際化文件*/(value = "${spring.messages.basename:message}")private String basename;public void init() {logger.info("init MessageResourceExtension...");if (!StringUtils.isEmpty(baseFolder)) {try {this.setBasenames(getAllBaseNames(baseFolder));} catch (IOException e) {logger.error(e.getMessage());}}//設(shè)置父MessageSourceResourceBundleMessageSource parent = new ResourceBundleMessageSource();parent.setBasename(basename);this.setParentMessageSource(parent);}/*** 獲取文件夾下所有的國(guó)際化文件名** @param folderName 文件名* @return* @throws IOException*/private String[] getAllBaseNames(String folderName) throws IOException {Resource resource = new ClassPathResource(folderName);File file = resource.getFile();List<String> baseNames = new ArrayList<>();if (file.exists() && file.isDirectory()) {this.getAllFile(baseNames, file, "");} else {logger.error("指定的baseFile不存在或者不是文件夾");}return baseNames.toArray(new String[baseNames.size()]);}/*** 遍歷所有文件** @param basenames* @param folder* @param path*/private void getAllFile(List<String> basenames, File folder, String path) {if (folder.isDirectory()) {for (File file : folder.listFiles()) {this.getAllFile(basenames, file, path + folder.getName() + File.separator);}} else {String i18Name = this.getI18FileName(path + folder.getName());if (!basenames.contains(i18Name)) {basenames.add(i18Name);}}}/*** 把普通文件名轉(zhuǎn)換成國(guó)際化文件名** @param filename* @return*/private String getI18FileName(String filename) {filename = filename.replace(".properties", "");for (int i = 0; i < 2; i++) {int index = filename.lastIndexOf("_");if (index != -1) {filename = filename.substring(0, index);}}return filename;} } 復(fù)制代碼依次解釋一下幾個(gè)方法。
所以簡(jiǎn)單來(lái)說(shuō)就是在MessageResourceExtension被實(shí)例化之后,把'i18n'文件夾下的資源文件的名字,加載到Basenames中。現(xiàn)在來(lái)看一下效果。
首先我們?cè)赼pplication.properties文件中添加一個(gè)spring.messages.baseFolder=i18n,這會(huì)把'i18n'這個(gè)值賦值給MessageResourceExtension中的baseFolder。
在啟動(dòng)后看到控制臺(tái)里打印出了init信息,表示被@PostConstruct注解的init()方法已經(jīng)執(zhí)行。
然后我們?cè)賱?chuàng)建兩組國(guó)際化信息文件:'dashboard'和'merchant',里面分別只有一個(gè)國(guó)際化信息:'dashboard.hello'和'merchant.hello'。
之后再修改一下hello.html文件,然后訪問(wèn)hello頁(yè)面。
... <body> <h1>國(guó)際化頁(yè)面!</h1> <p th:text="#{hello}"></p> <p th:text="#{merchant.hello}"></p> <p th:text="#{dashboard.hello}"></p> </body> ... 復(fù)制代碼可以看到網(wǎng)頁(yè)中加載了'message','dashboard'和'merchant'中的國(guó)際化信息,說(shuō)明我們已經(jīng)成功一次性加載了'i18n'文件夾下的文件。
后臺(tái)設(shè)置前端頁(yè)面顯示國(guó)際化信息的文件
s剛才那一節(jié)我們成功加載了多個(gè)國(guó)際化文件并顯示出了他們的國(guó)際化信息。但是'dashboard.properties'中的國(guó)際化信息為'dashboard.hello'而'merchant.properties'中的是'merchant.hello',這樣每個(gè)都要寫一個(gè)前綴豈不是很麻煩,現(xiàn)在我想要在'dashboard'和'merchant'的國(guó)際化文件中都只寫'hello'但是顯示的是'dashboard'或'merchant'中的國(guó)際化信息。
在MessageResourceExtension重寫resolveCodeWithoutArguments方法(如果有字符格式化的需求就重寫resolveCode方法)。
("messageSource") public class MessageResourceExtension extends ResourceBundleMessageSource {...public static String I18N_ATTRIBUTE = "i18n_attribute";protected String resolveCodeWithoutArguments(String code, Locale locale) {// 獲取request中設(shè)置的指定國(guó)際化文件名ServletRequestAttributes attr = (ServletRequestAttributes) RequestContextHolder.currentRequestAttributes();final String i18File = (String) attr.getAttribute(I18N_ATTRIBUTE, RequestAttributes.SCOPE_REQUEST);if (!StringUtils.isEmpty(i18File)) {//獲取在basenameSet中匹配的國(guó)際化文件名String basename = getBasenameSet().stream().filter(name -> StringUtils.endsWithIgnoreCase(name, i18File)).findFirst().orElse(null);if (!StringUtils.isEmpty(basename)) {//得到指定的國(guó)際化文件資源ResourceBundle bundle = getResourceBundle(basename, locale);if (bundle != null) {return getStringOrNull(bundle, code);}}}//如果指定i18文件夾中沒(méi)有該國(guó)際化字段,返回null會(huì)在ParentMessageSource中查找return null;}... } 復(fù)制代碼在我們重寫的resolveCodeWithoutArguments方法中,從HttpServletRequest中獲取到‘I18N_ATTRIBUTE’(等下再說(shuō)這個(gè)在哪里設(shè)置),這個(gè)對(duì)應(yīng)我們想要顯示的國(guó)際化文件名,然后我們?cè)贐asenameSet中查找該文件,再通過(guò)getResourceBundle獲取到資源,最后再getStringOrNull獲取到對(duì)應(yīng)的國(guó)際化信息。
現(xiàn)在我們到我們的HelloController里加兩個(gè)方法。
public class HelloController {("/hello")public String index(HttpServletRequest request) {request.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, "hello");return "system/hello";}("/dashboard")public String dashboard(HttpServletRequest request) {request.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, "dashboard");return "dashboard";}("/merchant")public String merchant(HttpServletRequest request) {request.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, "merchant");return "merchant";} } 復(fù)制代碼看到我們?cè)诿總€(gè)方法中都設(shè)置一個(gè)對(duì)應(yīng)的'I18N_ATTRIBUTE',這會(huì)在每次請(qǐng)求中設(shè)置對(duì)應(yīng)的國(guó)際化文件,然后在MessageResourceExtension中獲取。
這時(shí)我們看一下我們的國(guó)際化文件,我們可以看到所有關(guān)鍵字都是'hello',但是信息卻不同。
同時(shí)新增兩個(gè)html文件分別是'dashboard.html'和'merchant.html',里面只有一個(gè)'hello'的國(guó)際化信息和用于區(qū)分的標(biāo)題。
<!-- 這是hello.html --> <body> <h1>國(guó)際化頁(yè)面!</h1> <p th:text="#{hello}"></p> </body> 復(fù)制代碼<!-- 這是dashboard.html --> <body> <h1>國(guó)際化頁(yè)面(dashboard)!</h1> <p th:text="#{hello}"></p> </body> 復(fù)制代碼<!-- 這是merchant.html --> <body> <h1>國(guó)際化頁(yè)面(merchant)!</h1> <p th:text="#{hello}"></p> </body> 復(fù)制代碼這時(shí)我們啟動(dòng)項(xiàng)目看一下。
可以看到雖然在每個(gè)頁(yè)面的國(guó)際化詞都是'hello',但是我們?cè)趯?duì)應(yīng)的頁(yè)面顯示了我們想要顯示的信息。
利用攔截器和注解自動(dòng)設(shè)置前端頁(yè)面顯示國(guó)際化信息的文件
雖然已經(jīng)可以指定對(duì)應(yīng)的國(guó)際化信息,但是這樣要在每個(gè)controller里的HttpServletRequest中設(shè)置國(guó)際化文件實(shí)在太麻煩了,所以現(xiàn)在我們實(shí)現(xiàn)自動(dòng)判定來(lái)顯示對(duì)應(yīng)的文件。
首先我們創(chuàng)建一個(gè)注解,這個(gè)注解可以放在類上或者方法上。
({ElementType.TYPE, ElementType.METHOD}) (RetentionPolicy.RUNTIME) public I18n {/*** 國(guó)際化文件名*/String value(); } 復(fù)制代碼然后我們把這個(gè)創(chuàng)建的I18n注解放在剛才的Controller方法中,為了顯示他的效果,我們?cè)賱?chuàng)建一個(gè)ShopController和UserController,同時(shí)也創(chuàng)建對(duì)應(yīng)的'shop'和'user'的國(guó)際化文件,內(nèi)容也都是一個(gè)'hello'。
public class HelloController {("/hello")public String index() {return "system/hello";}18n("dashboard")("/dashboard")public String dashboard() {return "dashboard";}18n("merchant")("/merchant")public String merchant() {return "merchant";} } 復(fù)制代碼18n("shop") public class ShopController {("shop")public String shop() {return "shop";} } 復(fù)制代碼 public class UserController {("user")public String user() {return "user";} } 復(fù)制代碼我們把I18n注解分別放在HelloController下的dashboard和merchant方法下,和ShopController類上。并且去除了原來(lái)dashboard和merchant方法下設(shè)置‘I18N_ATTRIBUTE’的語(yǔ)句。
準(zhǔn)備工作都做好了,現(xiàn)在看看如何實(shí)現(xiàn)根據(jù)這些注解自動(dòng)的指定國(guó)際化文件。
public class MessageResourceInterceptor implements HandlerInterceptor {public void postHandle(HttpServletRequest req, HttpServletResponse rep, Object handler, ModelAndView modelAndView) {// 在方法中設(shè)置i18路徑if (null != req.getAttribute(MessageResourceExtension.I18N_ATTRIBUTE)) {return;}HandlerMethod method = (HandlerMethod) handler;// 在method上注解了i18I18n i18nMethod = method.getMethodAnnotation(I18n.class);if (null != i18nMethod) {req.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, i18nMethod.value());return;}// 在Controller上注解了i18I18n i18nController = method.getBeanType().getAnnotation(I18n.class);if (null != i18nController) {req.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, i18nController.value());return;}// 根據(jù)Controller名字設(shè)置i18String controller = method.getBeanType().getName();int index = controller.lastIndexOf(".");if (index != -1) {controller = controller.substring(index + 1, controller.length());}index = controller.toUpperCase().indexOf("CONTROLLER");if (index != -1) {controller = controller.substring(0, index);}req.setAttribute(MessageResourceExtension.I18N_ATTRIBUTE, controller);}public boolean preHandle(HttpServletRequest req, HttpServletResponse rep, Object handler) {// 在跳轉(zhuǎn)到該方法先清除request中的國(guó)際化信息req.removeAttribute(MessageResourceExtension.I18N_ATTRIBUTE);return true;} } 復(fù)制代碼簡(jiǎn)單講解一下這個(gè)攔截器。
首先,如果request中已經(jīng)有'I18N_ATTRIBUTE',說(shuō)明在Controller的方法中指定設(shè)置了,就不再判斷。
然后判斷一下進(jìn)入攔截器的方法上有沒(méi)有I18n的注解,如果有就設(shè)置'I18N_ATTRIBUTE'到request中并退出攔截器,如果沒(méi)有就繼續(xù)。
再判斷進(jìn)入攔截的類上有沒(méi)有I18n的注解,如果有就設(shè)置'I18N_ATTRIBUTE'到request中并退出攔截器,如果沒(méi)有就繼續(xù)。
最后假如方法和類上都沒(méi)有I18n的注解,那我們可以根據(jù)Controller名自動(dòng)設(shè)置指定的國(guó)際化文件,比如'UserController'那么就會(huì)去找'user'的國(guó)際化文件。
攔截器完成了,現(xiàn)在把攔截器配置到系統(tǒng)中。修改I18nApplication啟動(dòng)類:
public class I18nApplication {...public WebMvcConfigurer webMvcConfigurer() {return new WebMvcConfigurer() {//攔截器public void addInterceptors(InterceptorRegistry registry) {registry.addInterceptor(new LocaleChangeInterceptor()).addPathPatterns("/**");registry.addInterceptor(new MessageResourceInterceptor()).addPathPatterns("/**");}};} } 復(fù)制代碼現(xiàn)在我們?cè)龠\(yùn)行一下看看效果,看到每個(gè)鏈接都顯示的他們對(duì)應(yīng)的國(guó)際化信息里的內(nèi)容。
最后
剛才完成了我們整個(gè)國(guó)際化增強(qiáng)的基本功能,最后我把全部代碼整理了一下,并且整合了bootstrap4來(lái)展示了一下功能的實(shí)現(xiàn)效果。
詳細(xì)的代碼可以看我Github上Spring-Boot-I18n-Pro的代碼
原文地址:zzzzbw.cn
2018/8/30更新
文章發(fā)布后,有人向我提到當(dāng)把項(xiàng)目打成jar包之后執(zhí)行java -jar i18n-0.0.1.jar的方式來(lái)運(yùn)行程序會(huì)報(bào)錯(cuò)。看到這樣的反饋我立刻就意識(shí)到,確實(shí)在讀取i18n的國(guó)際化文件的時(shí)候用的是File的形式來(lái)讀取文件名的,假如打包成jar包后所有文件都是在壓縮文件夾中,就不能簡(jiǎn)單的以File的形式來(lái)獲取到文件夾下的所有文件了。因?yàn)楣镜捻?xiàng)目是以war包的形式在Tomcat下運(yùn)行,所以沒(méi)有發(fā)現(xiàn)這個(gè)問(wèn)題。
主要問(wèn)題是在MessageResourceExtension類在spring-boot啟動(dòng)時(shí)讀取配置文件導(dǎo)致的,所以修改MessageResourceExtension:
("messageSource") public class MessageResourceExtension extends ResourceBundleMessageSource {.../*** 獲取文件夾下所有的國(guó)際化文件名*/private String[] getAllBaseNames(final String folderName) throws IOException {URL url = Thread.currentThread().getContextClassLoader().getResource(folderName);if (null == url) {throw new RuntimeException("無(wú)法獲取資源文件路徑");}List<String> baseNames = new ArrayList<>();if (url.getProtocol().equalsIgnoreCase("file")) {// 文件夾形式,用File獲取資源路徑File file = new File(url.getFile());if (file.exists() && file.isDirectory()) {baseNames = Files.walk(file.toPath()).filter(path -> path.toFile().isFile()).map(Path::toString).map(path -> path.substring(path.indexOf(folderName))).map(this::getI18FileName).distinct().collect(Collectors.toList());} else {logger.error("指定的baseFile不存在或者不是文件夾");}} else if (url.getProtocol().equalsIgnoreCase("jar")) {// jar包形式,用JarEntry獲取資源路徑String jarPath = url.getFile().substring(url.getFile().indexOf(":") + 2, url.getFile().indexOf("!"));JarFile jarFile = new JarFile(new File(jarPath));List<String> baseJars = jarFile.stream().map(ZipEntry::toString).filter(jar -> jar.endsWith(folderName + "/")).collect(Collectors.toList());if (baseJars.isEmpty()) {logger.info("不存在{}資源文件夾", folderName);return new String[0];}baseNames = jarFile.stream().map(ZipEntry::toString).filter(jar -> baseJars.stream().anyMatch(jar::startsWith)).filter(jar -> jar.endsWith(".properties")).map(jar -> jar.substring(jar.indexOf(folderName))).map(this::getI18FileName).distinct().collect(Collectors.toList());}return baseNames.toArray(new String[0]);}/*** 把普通文件名轉(zhuǎn)換成國(guó)際化文件名*/private String getI18FileName(String filename) {filename = filename.replace(".properties", "");for (int i = 0; i < 2; i++) {int index = filename.lastIndexOf("_");if (index != -1) {filename = filename.substring(0, index);}}return filename.replace("\\", "/");}... } 復(fù)制代碼在getAllBaseNames()方法中會(huì)先判斷項(xiàng)目的Url形式為文件形式還是jar包形式。
如果是文件形式則就以普通文件夾的方式讀取,這里還用了java8中的Files.walk()方法獲取到文件夾下的所有文件,比原來(lái)自己寫遞歸來(lái)讀取方便多了。
如果是jar包的形式,那么就要用JarEntry來(lái)處理文件了。
首先是獲取到項(xiàng)目jar包所在的的目錄,如E:/workspace/java/Spring-Boot-I18n-Pro/target/i18n-0.0.1.jar這種,然后根據(jù)該目錄new一個(gè)JarFile。
接著遍歷這個(gè)JarFile包下的資源,這會(huì)把我們項(xiàng)目jar包下的所有文件都讀取出來(lái),所以我們要先找到我們i18n資源文件所在的目錄,通過(guò).filter(jar -> jar.endsWith(folderName + "/"))獲取資源所在目錄。
接下來(lái)就是判斷JarFile包下的文件是否在i18n資源目錄了,如果是則調(diào)用getI18FileName()方法將其格式化成我們所需要的名字形式。
經(jīng)過(guò)這段操作就實(shí)現(xiàn)了獲取jar包下i18n的資源文件名了。
總結(jié)
以上是生活随笔為你收集整理的自己动手在Spring-Boot上加强国际化功能的全部?jī)?nèi)容,希望文章能夠幫你解決所遇到的問(wèn)題。
- 上一篇: 异地取数
- 下一篇: linux以太坊开发环境搭建