本文主要介绍SpringBoot:Thymeleaf模板引擎和国际化的使用
一、SpringBoot:Thymeleaf模板引擎
在学习Thymeleaf之前,我们回忆一下JSP:
JSP(全称JavaServer Pages)是由Sun Microsystems公司主导创建的一种动态网页技术标准。JSP部署于网络服务器上,可以响应客户端发送的请求,并根据请求内容动态地生成HTML、XML或其他格式文档的Web网页,然后返回给请求者。JSP技术以Java语言作为脚本语言,为用户的HTTP请求提供服务,并能与服务器上的其它Java程序共同处理复杂的业务需求。----摘自百度百科
JSP最大的特点,是支持Java开发,能以模板话的方式,动态添加动态网页内容,轻松实现数据显示与交互。
然而,SpringBoot项目以jar的形式发布,而且内嵌Tomcat,它默认是不支持JSP的。
- SpringBoot推荐使用模板引擎:
其实jsp就是一个模板引擎,还有用的比较多的freemarker,包括SpringBoot给我们推荐的Thymeleaf。
Thymeleaf官网
在知乎上,有很多大神解答:Spring / SpringBoot为什么推荐使用thymeleaf?
- 众多模板引擎,其实它们的思想都是大同小异:
模板引擎重要的是交互数据,能够动态的将页面数据展示,后台分装的数据要如何与前端的所需要的数据对应起来,模板引擎在中间起到定位传递数据的效果。
- 介绍SpringBoot如何使用Thmeleaf
Thymeleaf 官网:https://www.thymeleaf.org/
Thymeleaf 在Github 的主页:https://github.com/thymeleaf/thymeleaf
1.引入Thymeleaf
在Spring官方文档中,可以找到Guies:Learn how to create a web page with Spring MVC and Thymeleaf.
在SpringBoot文档可以找到Thymeleaf的Starter:https://docs.spring.io/spring-boot/docs/2.4.4/reference/html/using-spring-boot.html#using-boot-starter
找到对应的pom依赖:
<!--Thymeleaf--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-thymeleaf</artifactId> </dependency>
Maven会自动下载jar包:
- 引入完毕,如何使用:
首先得按照SpringBoot的自动配置原理看一下我们这个Thymeleaf的自动配置规则,在按照那个规则,我们进行使用。
我们去找一下Thymeleaf的自动配置类:ThymeleafProperties (看到其默认的前缀/后缀)
使用Thymeleaf什么都不需要配置,我们将文件放到指定的文件夹下即可,Thymeleaf会帮助我们自动渲染。非常符合开箱即用的原则,而且在无网络的状态下,依然可以使用,代码风格优雅,简洁。
- 测试
1.编写一个TestController
import org.springframework.stereotype.Controller; import org.springframework.web.bind.annotation.RequestMapping; @Controller public class TestController { @RequestMapping("/test") public String test1(){ //classpath:/templates/test.html return "test"; } }
2、编写一个测试页面 test.html 放在 templates 目录下
<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <span style="background: orange">Hello,Thymeleaf!</span> </body> </html>
3.Run
- Thymeleaf 语法学习
Thymeleaf 官网:https://www.thymeleaf.org/
- 做个简单的练习 :需求查出一些数据,并在页面中展示。
1、修改测试请求,增加数据传输
@RequestMapping("test2") public String test2(Model model){ //存入数据 String name="I love China"; model.addAttribute("msg",name); return "test2"; }
2、要使用thymeleaf,需要在html文件中导入命名空间的约束,方便提示。
xmlns:th="http://www.thymeleaf.org"
3.编写前端页面
<!DOCTYPE html> <html lang="en" xmlns:th="http://www.thymeleaf.org"> <head> <meta charset="UTF-8"> <title>你来啦</title> </head> <body> <h4>注意:要使用thymeleaf,需要在html文件中导入命名空间的约束,方便提示!</h4> <h2 th:text="${msg}"></h2> </body> </html>
4.Run
- Thymeleaf的使用语法:
1.官方在线文档PDF
2.官方在线参考文档
- 文档需要自己看,下面再来个小测试:
1.Controller,放一些数据
@RequestMapping("/test3") public String test3(Map<String,Object> map){ map.put("msg","<h1>Hello</h1>"); map.put("users", Arrays.asList("zhouzhou",new Date())); return "test3"; }
2.测试页面取出数据:
<!DOCTYPE html> <html lang="en" xmlns:th="http://www.thymeleaf.org"> <head> <meta charset="UTF-8"> <title>My Thymeleaf</title> </head> <body> <h1>多看文档,了解更多Thymeleaf的使用</h1> <br/> <h2>用th:text不会解析html,用th:utext会解析html,在页面中显示相应的样式</h2> <div th:text="${msg}"></div> <div th:utext="${msg}"></div> ==================分割线=============================== <!--遍历数据--> <h4 th:each="user:${users}" th:text="${user}"></h4> <h4> <!--行内写法--> <span th:each="user:${users}">[[${user}]]</span> </h4> </body> </html>
3.Run
二、SpringBoot:国际化
有些网站会去涉及中英文甚至多语言的切换——国际化
- 准备工作
先在IDEA中统一设置properties的编码问题:
- 编写国际化配置文件
1.在resources资源文件下新建一个i18n目录,存放国际化配置文件
2.建立一个login.properties文件、login_zh_CN.properties文件,发现IDEA自动识别了我们要做国际化操作。
文件夹变了!
3.利用这里,再新建一个文件
显示:
4.编写配置也可以用视图:
5.在视图中,点击 + 号就可以直接添加属性
6.新建一个login.tip,可以看到边上有三个文件框可以输入
7.添加内容
8.抽取页面需要显示的国际化页面消息,找到哪些内容需要编写国际化的配置
找到配置文件并查看:
配置文件部分搞定!
9.由于我们添置了i18n目录,所以我们要去配置这个messages的路径:
spring.messages.basename=i18n.login
- 配置文件生效探究
看一下SpringBoot对国际化的自动配置!这里又涉及到一个类:MessageSourceAutoConfiguration
/** * {@link EnableAutoConfiguration Auto-configuration} for {@link MessageSource}. * * @author Dave Syer * @author Phillip Webb * @author Eddú Meléndez * @since 1.5.0 */ @Configuration(proxyBeanMethods = false) @ConditionalOnMissingBean(name = AbstractApplicationContext.MESSAGE_SOURCE_BEAN_NAME, search = SearchStrategy.CURRENT) @AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE) @Conditional(ResourceBundleCondition.class) @EnableConfigurationProperties public class MessageSourceAutoConfiguration { private static final Resource[] NO_RESOURCES = {}; @Bean @ConfigurationProperties(prefix = "spring.messages") public MessageSourceProperties messageSourceProperties() { return new MessageSourceProperties(); } //获取properties 传递过来的值,进行判断 @Bean public MessageSource messageSource(MessageSourceProperties properties) { ResourceBundleMessageSource messageSource = new ResourceBundleMessageSource(); // 设置国际化文件的基础名(去掉语言国家代码的) if (StringUtils.hasText(properties.getBasename())) { messageSource.setBasenames(StringUtils .commaDelimitedListToStringArray(StringUtils.trimAllWhitespace(properties.getBasename()))); } if (properties.getEncoding() != null) { messageSource.setDefaultEncoding(properties.getEncoding().name()); } messageSource.setFallbackToSystemLocale(properties.isFallbackToSystemLocale()); Duration cacheDuration = properties.getCacheDuration(); if (cacheDuration != null) { messageSource.setCacheMillis(cacheDuration.toMillis()); } messageSource.setAlwaysUseMessageFormat(properties.isAlwaysUseMessageFormat()); messageSource.setUseCodeAsDefaultMessage(properties.isUseCodeAsDefaultMessage()); return messageSource; } //省略下面代码
- 测试1:设计一个简单的页面
<!DOCTYPE html> <html lang="en" xmlns:th="http://www.thymeleaf.org"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="/test"> <h3 th:text="#{login.tip}">Please Sign in</h3> <input type="text" th:placeholder="#{login.username}"> <input type="password" th:placeholder="#{login.password}"> <a>中文</a> <a>English</a> </form> </body> </html>
Run: 启动项目,访问一下,发现已经自动识别为中文。
- 测试2:实现根据按钮自动切换中文英文
配置国际化解析
在Spring中有一个国际化的Locale (区域信息对象);里面有一个叫做LocaleResolver (获取区域信息对象)的解析器。
我们去webmvc自动配置文件,寻找SpringBoot默认配置:
其中,FIXED常量源码:
另外查看到,AcceptHeaderLocaleResolver这个类中有一个方法:
所以,想让我们的国际化资源生效,就需要让我们自己的Locale生效!我们写一个自己的LocaleResolver,可以在链接上携带区域信息!
1.修改一下前端页面的跳转连接:
<!DOCTYPE html> <html lang="en" xmlns:th="http://www.thymeleaf.org"> <head> <meta charset="UTF-8"> <title>Title</title> </head> <body> <form action="/test"> <h3 th:text="#{login.tip}">Please Sign in</h3> <input type="text" th:placeholder="#{login.username}"> <input type="password" th:placeholder="#{login.password}"> <!-- 这里传入参数不需要使用问号(?),使用(key=value)--> <a th:href="@{/test4(language='zh_CN')}">中文</a> <a th:href="@{/test4(language='en_US')}">English</a> </form> </body> </html>
2.添加组件(处理国际化需求)
package com.zhou.component; import org.springframework.util.ObjectUtils; import org.springframework.web.servlet.LocaleResolver; import javax.servlet.http.HttpServletRequest; import javax.servlet.http.HttpServletResponse; import java.util.Locale; public class MyLocaleResolver implements LocaleResolver { //解析请求 @Override public Locale resolveLocale(HttpServletRequest httpServletRequest) { String language = httpServletRequest.getParameter("language"); Locale aDefault = Locale.getDefault();//如果没有获取到就是用系统默认的 //如果请求连接不为空:StringUtils.isEmpty(language)被弃用了。 if(!ObjectUtils.isEmpty(language)){ //分割请求参数 String[] split = language.split("_"); //国家,地区 aDefault = new Locale(split[0], split[1]); } return aDefault; } @Override public void setLocale(HttpServletRequest httpServletRequest, HttpServletResponse httpServletResponse, Locale locale) { } }
3.配置组件
package com.zhou.config; import com.zhou.component.MyLocaleResolver; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; import org.springframework.web.servlet.LocaleResolver; import org.springframework.web.servlet.config.annotation.WebMvcConfigurer; //扩展SpringBoot @Configuration public class MyMvcConfig implements WebMvcConfigurer { //自定义国际化组件生效 @Bean public LocaleResolver localeResolver(){ return new MyLocaleResolver(); } }
4.测试:
访问:http://localhost:8084/test4 并点击:中文 或者English
点击中文:
点击English