官方手册: http://www.thymeleaf.org/doc/tutorials/3.0/usingthymeleaf.html
Thymeleaf是SpringBoot官方所推荐使用的页面模板语言
优点:
动静结合:Thymeleaf 在有网络和无网络的环境下皆可运行,即它可以让美工在浏览器查看页面的静态效果,也可以让程序员在服务器查看带数据的动态页面效果。这是由于它支持 html 原型,然后在 html 标签里增加额外的属性来达到模板+数据的展示方式。浏览器解释 html 时会忽略未定义的标签属性,所以 thymeleaf 的模板可以静态地运行;当有数据返回到页面时,Thymeleaf 标签会动态地替换掉静态内容,使页面动态显示。
开箱即用:它提供标准和spring标准两种方言,可以直接套用模板实现JSTL、 OGNL表达式效果,避免每天套模板、改jstl、改标签的困扰。同时开发人员也可以扩展和创建自定义的方言。
多方言支持:Thymeleaf 提供spring标准方言和一个与 SpringMVC 完美集成的可选模块,可以快速的实现表单绑定、属性编辑器、国际化等功能。
与SpringBoot完美整合,SpringBoot提供了Thymeleaf的默认配置,并且为Thymeleaf设置了视图解析器,我们可以像以前操作jsp一样来操作Thymeleaf。代码几乎没有任何区别,就是在模板语法上有区别。
<!--SpringBoot 集成 Thymeleaf 的起步依赖-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>
spring.thymeleaf.cache=false
# 前缀:
spring.thymeleaf.prefix=classpath:/templates/
# 后缀:
spring.thymeleaf.suffix=.html
thymeleaf
对.html
的内容要求很严格,建议增加下面的配置:#去掉html5语法验证
spring.thymeleaf.mode=LEGACYHTML5
spring.thymeleaf.mode
的默认值是HTML5
,其实是一个很严格的检查,改为LEGACYHTML5
可以得到一个可能更友好亲切的格式要求。
使用该配置时,需要额外引入NekoHTML
库,相关依赖:
<dependency>
<groupId>net.sourceforge.nekohtml</groupId>
<artifactId>nekohtml</artifactId>
</dependency>
通过Model对象将后台的值传递回前台,Model对象实际上是将后台的数据放到了request作用域中,也可以使用HttpServletRequest对象进行传值,但是在idea中会显示报错,且没有提示,但是值可以正常取出来。推荐使用Model对象
@Controller
public class ThymeleafController {
@RequestMapping("/leaf")
public String leaf(Model model) {
model.addAttribute("data", "SpringBoot集成Thymeleaf");
return "aa";
}
}
Springboot 使 用 thymeleaf 作 为 视 图 展 示 , 约 定 将 模 板 文 件 放 置 在 src/main/resource/templates
目录下,静态资源放置在 src/main/resource/static
目录下。
在templates新建一个普通的HTML5文件,然后在html标签中添加属性:xmlns:th="http://www.thymeleaf.org"
就可以了
<html lang="en" xmlns:th="http://www.thymeleaf.org"></html>
<span th:text="${user.id}"></span>
Thymeleaf中所有的表达式都需要写在"指令"中,指令是HTML5中的自定义属性,在Thymeleaf中所有指令都是以
th:
开头。表达式${user.name}是写在自定义属性中,因此在静态环境下,表达式的内容会被当做是普通字符串,浏览器会自动忽略这些指令,这样就不会报错了。
假设 request 作用域中存在键值对:"key"=value
,可以使用
使用标准变量表达式:
<span th:text="${#httpServletRequest.getAttribute('key')}"></span>
或者:<span th:text="${key}"></span>
假设session 作用域中存在键值对:"key"=value
使用标准变量表达式需要加个前缀 session
:
<span th:text="${session.key}"></span>
假设 application 作用域中存在键值对:"key"=value
使用标准变量表达式需要加个前缀 application
:
<span th:text="${application.key}"></span>
语法:${对象名?.属性名}
如果一个对象可能为 null 时,并且要获取该对象的属性时,这时需要在该对象后面加一个"?
",当一个为 null 的对象去获取一个属性时,会报TemplateInputException
(模板输入/解析异常)
th:object
属性来绑定对象;也可以不使用 th:object
属性来绑定对象,则使用方法与${}相同<div th:object="${user}">
用户编号:<span th:text="*{id}"></span><br/>
用户姓名:<span th:text="*{name}"></span><br/>
</div>
<script src="...">
、 <link href="...">
、 <a href="...">
、 <form action="...">
、 <img src="">
等,可以在 URL 路径中动态获取数据在 URL 表达式前加/
,会自动加上上下文根,即上下文相关。
绝对的URL,比如 http://www.thymeleaf.org
相对URL,可以是:
- 页面相对,像 user/login.html
- 上下文相关,如/itemdetails?id=3(服务器中的上下文名称将自动添加)
- 与服务器相关,~/billing/processInvoice(允许在同一服务器中调用另一个上下文(=应用程序)中的URL)。
- 协议相对URL,如 //code.jquery.com/jquery-2.0.3.min.js
<h1>绝对路径</h1>
<a th:href="@{http://localhost:8080/thymeleaf/info}">绝对路径</a>
<a th:href="@{'http://localhost:8080/thymeleaf/user/info?id='+${user.id}}">绝对路径(带参数)</a>
<h1>相对路径,常用</h1>
<a th:href="@{/thymeleaf/info}">相对路径</a>
<a th:href="@{'/thymeleaf/user/info?id=' + ${user.id}}">相对路径(带参数)(字符串拼接) </a>
<a th:href="@{/thymeleaf/info(id=${user.id}, name=${user.name})}">推荐:带多个参数路径写法</a>
因为我们 Thymeleaf 是以 html 为载体的,所以 html 不会认识${}语法。
我们请求的流程是,发送请求给服务器,服务器接收请求后,处理请求,跳转到指定的静态 html 页面,在服务器端, Thymeleaf 模板引擎会按照它的语法, 对动态数据进行处理。所以如果要是 th
开头,模板引擎能够识别,会在服务器端进行处理,获取数据;如果没有以 th
开头,那么 Thymeleaf 模板引擎不会处理,直接返回给客户端了。
th:action
:定义后台控制器的路径,类似<form>
标签的 action 属性,主要结合 URL 表达式,获取动态变量
设置请求方法
定义超链接, 主要结合 URL 表达式,获取动态变量
用于外部资源引入,比如<script>
标签的 src 属性, <img>
标签的 src 属性,常与@{}表达式结合使用。
SpringBoot 项目的静态资源都放到 resources 的 static 目录下。放到 static 路径下的内容,写路径时不需要写上 static
<!--以下方式无法引入 js-->
<script src="/static/js/jquery-1.7.2.min.js"></script>
<!--该方法是常用方法-->
<script type="text/javascript" th:src="@{/jquery-1.7.2.min.js}"></script>
替换 html 标签中的 id,name,value 属性
该属性也是用于给 HTML 中某元素的某属性赋值,好处是可以给 html 中没有定义的属性动态的赋值。
<span zhangsan="${user.name}">该方法赋值不成功</span>
<!--通过 th:attr 对自定义的属性赋值-->
<span th:attr="zhangsan=${user.name}">可以赋值成功</span>
用于文本的显示,该属性显示的文本在标签体中,即两个标签对之间。如果是表单标签,使用th:value
用于数据对象绑定,通常用于选择变量表达式(星号表达式)
目前 thymeleaf 版本要求只能传递数字和布尔值
<!--目前 thymeleaf 版本要求只能传递数字和布尔值-->
<a th:onclick="'show(' + ${user.id} + ')'">点击:显示学生编号</a>
<script type="text/javascript">
function show(id) {
alert("用户编号为: " + id);
}
</script>
内敛文本表达式不依赖于 html 标签,直接使用内敛表达式[[表达式]]
即可获取动态数据(相当于是加了两个方括号的EL表达式了), 但必须要求在父级标签上加 th:inline = “text”
属性
注意:一般我们将内联文本放到<body th:inline="text">
标签中
<div th:inline="text">
用户编号: <div>[[${user.id}]]</div><br/>
用户姓名: [[${user.name}]]<br/>
</div>
th:inline="javascript"
用于 js 代码中获取后台的动态数据
<script type="text/javascript" th:inline="javascript">
function showInlineJavaScript() {
alert("我叫 " + [[${user.name}]] + " 联系方式: " + [[${user.phone}]]);
}
</script>
注意:在内敛脚本中使用可能为 null 的对象去获取属性时,一定要使用${对象名.属性名}
,即使先进行的判空处理(在这踩坑了)。如:
<script type="text/javascript" th:inline="javascript">
if([[${user != null}]]) {
//虽然先进行的判断,user不为null,但是thymeleaf在解析内敛脚本时,与html代码不同,是不会跳过代码的,所以user.name也会正常解析
//当user为null时,就会报TemplateInputException
alert("我叫 " + [[${user?.name}]] + " 联系方式: " + [[${user?.phone}]]);
}
</script>
用于在页面中指定一个通用(复用)的代码片段,例如在页面 common/view1.html
中有如下定义:
<div th:fragment="pageHead">
一些代码,比如说同一个网站每个页面的头和尾都相同,将公共的代码抽取出来写在这里
</div>
在页面中引入 th:fragment 定义的代码片段,主要有下面三种形式:
th:inclued="view::selector"
:"::
"前面是模板文件名,后面是选择器::selector
:只写选择器,这里指fragment名称,则加载本页面对应的fragmentview
或者"view::html"
:只写模板文件名或者使用 html 标签,则加载整个页面比如说,加载(1)中 th:fragment 定义的代码片段:
<div th:include="common/view1::pageHead"></div>
th:replace 和 th:include 都是加载代码块内容,但是还是有所不同
th:include
:加载模板的内容: 读取加载节点的内容(不含节点名称),替换div内容
th:replace
:替换当前标签为模板中的标签,加载的节点会整个替换掉加载它的div
例如:
<!-- th:fragment 定义用于加载的块 -->
<span th:fragment="view">
这是公共部分
</span>
引用时如下:
include:
<div th:include="pagination::view">1</div>
replace:
<div th:replace="pagination::view">2</div>
结果如下:
include:
<div>这是公共部分</div>
replace:
<span>这是公共部分</span>
设置样式
设置CSS选择器
大部分的html标签都有thymeleaf相对应的属性,这里就不一一列举出来了,在IDEA中,打个th:
之后有提示的,拿过来使用即可
这个属性非常常用,比如从后台传来一个对象集合那么就可以使用此属性遍历输出,它与 JSTL 中的<c: forEach>类似,此属性既可以循环遍历集合,也可以循环遍历数组及 Map
th:each="user , iterStat : ${userList}"
或者th:each="user : ${userList}"
index
:当前迭代对象的 index(从 0 开始计算)count
:当前迭代对象的个数(从 1 开始计算) 这两个用的较多,
“隔开,第二个参数和第三个参数之间使用”:
"隔开<h1>遍历集合或者数组</h1>
<div th:each="user,userStat:${userList}">
<span th:text="${userStat.index}"></span>
<span th:text="${user.id}"></span>
<span th:text="${user.name}"></span>
</div>
<h1>遍历map,每个元素就是一个键值对,通过key和value来取出键或者值</h1>
<tr th:each="userMap:${userMaps}">
<td th:text="${userMap.key}"></td>
<td th:text="${userMap.value}"></td>
<td th:text="${userMap.value.name}"></td>
</tr>
th:if是条件满足则执行,th:unless相反,条件不满足时执行
<!--sex等于1则显示男,sex等于0则显示女-->
<div th:if="${sex eq 1}">
男: <input type="radio" name="sex" th:value="1"/>
</div>
<div th:if="${sex eq 0}">
女: <input type="radio" name="sex" th:value="0"/>
</div>
一旦某个 case 判断值为 true,剩余的 case 默认不执行,"*
"表示默认的 case,前面的 case 都不匹配时候,执行默认的 case
<h1>th:switch/th:case 用法</h1>
<div th:switch="${sex}">
<span th:case="1">性别:男</span><br/>
<span th:case="2">性别:女</span><br/>
<span th:case="*">性别:保密</span>
</div>
字面量:对应数据类型的合法取值,可以在 html 页面直接使用,不需要后台传递
'...'
包围的字符串为文本字面量<h1>文本字面量:用单引号'...'包围的字符串</h1>
<a th:href="@{'/user/info?id=' + ${user.id}}">文本字面的路径使用</a><br/>
<h1>数字字面量</h1>
<span th:text="${1949 + 70}">1949</span><br/>
<h1>boolean字面量</h1>
<div th:if="${success}">执行成功</div>
<h1>null 字面量</h1>
<span th:if="${user ne null}">用户不为空</span><br/>
除了使用加号拼接文本字面量之外,可以使用"|要拼接的内容|
"来进行拼接
<h1>文本字面量使用"+"拼接字符串</h1>
<span th:text="'共' + ${totalRows} + '条'+${totalPage} + '页,当前第' + ${currentPage} + '页'"></span>
<h1>使用"|要拼接的内容|"拼接字符串,最后的结果与上面的相同</h1>
<span th:text="|共${totalRows}条${totalPage}页,当前第${currentPage}页|"></span>
模板引擎提供了一组内置的对象,这些内置的对象可以直接在模板中使用,这些对象由"#
"号开始引用,我们比较常用的内置对象
#request
相 当 于java中 HttpServletRequest
对 象 , 这 是 3.x 版 本 , 若 是 2.x 版 本 使 用 #httpServletRequest
, 一般用于在页面获取应用的上下文根,一般在 js 中请求路径中加上可以避免 404
var scheme = [[${#request.getScheme()}]];
var serverName = [[${#request.getServerName()}]];
var serverPort = [[${#request.getServerPort()}]];
var contextPath = [[${#request.getContextPath()}]];
<script type="text/javascript" th:inline="javascript">
var basePath = [[${#httpServletRequest.getScheme() + "://" +
#httpServletRequest.getServerName() + ":" +
#httpServletRequest.getServerPort() +
#httpServletRequest.getContextPath()}]];
</script>
相当于java中的 HttpSession 对象,这是 3.x 版本,若是 2.x 版本使用#httpSession
<h1>从 SESSION 中获取用户名称</h1>
<span th:text="${#httpSession.getAttribute('username')}"></span>
模板引擎提供的一组功能性内置对象,可以在模板中直接使用这些对象提供的功能方法工作中常使用的数据类型,如集合,时间,数值,可以使用 Thymeleaf 的提供的功能性对象来处理它们
内置功能对象前都需要加#号,内置对象一般都以 s
结尾
#dates:java.util.Date 对象的实用方法:
<span th:text="${#dates.format(curDate, 'yyyy-MM-dd HH:mm:ss')}"></span>
#calendars:和 dates 类似, 但是 java.util.Calendar 对象;
#numbers:格式化数字对象的实用方法;
#strings:字符串对象的实用方法: contains,startsWith,prepending/appending,replace 等;
<span th:text="${#strings.replace('目标字符串', '被替换的字符串', '替换后的字符串')}"></span>
#objects:对 objects 操作的实用方法;
#bools:对布尔值求值的实用方法;
#arrays:数组的实用方法;
#lists:list 的实用方法,比如
<span th:text="${#lists.size(datas)}"></span>
#sets:set 的实用方法;
#maps:map 的实用方法;
#aggregates:对数组或集合创建聚合的实用方法;