1. 简介

Thymeleaf 是一个服务器端 Java 模板引擎,能够处理 HTML、XML、CSS、JAVASCRIPT 等模板文件。Thymeleaf 模板可以直接当作静态原型来使用,它主要目标是为开发者的开发工作流程带来优雅的自然模板,也是 Java 服务器端 HTML5 开发的理想选择

2. 创建模板文件

创建一个html模板文件:

1
2
3
4
5
6
7
8
9
10
<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
<head>
<title>Index Page</title>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
</head>
<body>
<p th:text="${message}">Welcome to BeiJing!</p>
</body>
</html>
  1. :声明文档的解析类型(document.compatMode),使用标准模式渲染页面,避免浏览器的怪异模式 (浏览器使用自己的怪异模式解析渲染页面).

标准模式,浏览器使用W3C的标准解析渲染页面

  1. : 引入 Thymeleaf 命名空间 (xmlns 属性规定文档的 xml 命名空间)

3、标准表达式语法

Thymeleaf 提供了非常丰富的标准表达式语法,总共有 8 大类:

  • 简单表达式
  • 字面值
  • 文本操作
  • 算术运算
  • 布尔运算
  • 比较和相等
  • 条件运算
  • 无操作符

3.1 简单表达式

语法 名称 描述 作用
${…} Variable Expressions 变量表达式 取出上下文变量的值
*{…} Selection Variable Expressions 选择表达式 取出选择的对象的属性值
#{…} Message Expressions 消息表达式 使文字消息国际化,I18N
@{…} Link URL Expressions 链接表达式 用于表示各种超链接地址
~{…} Fragment Expressions 片段表达式 引用一段公共的代码片段

3.1.1 ${…}   变量表达式

1
2
3
4
5
6
7
@GetMapping("/standard-expression-syntax/variables")
public String variables(ModelMap model, HttpSession session) {
model.put("now", new Date());
model.put("message", "Welcome to BeiJing!");
session.setAttribute("user", new User("fanlychie", "男", 24));
... ...
}

通过变量表达式${}取出上下文环境中的message变量:

1
<p th:text="${message}"></p>

3.1.2  *{…}   选择表达式

变量表达式${}是面向整个上下文的,而选择变量表达式*{}的上下文是父标签(th:object)所选择的对象

1
2
3
4
5
<div th:object="${session.user}">
<p th:text="*{name}"></p>
<p th:text="*{sex}"></p>
<p th:text="*{age}"></p>
</div>

3.1.3  #{…}   消息表达式

消息表达式可用于国际化文字信息.首先我们来了解一下 i18n 资源文件的命名规则:

  • basename.properties
  • basename_language.properties
  • basename_language_country.properties

basename是自定义的资源文件名称,languagecountry必须是 Java 支持的语言和国家。basename.properties是缺省加载的资源文件,当客户端根据本地语言查找不到相关的资源文件时,则使用该配置文件。

basename是自定义的资源文件名称,languagecountry必须是 Java 支持的语言和国家。basename.properties是缺省加载的资源文件,当客户端根据本地语言查找不到相关的资源文件时,则使用该配置文件。

创建文件src/main/resources/messages.properties

1
welcome.message = 北京欢迎你!

创建文件src/main/resources/messages_en_US.properties

1
welcome.message = Welcome to BeiJing!

在 IntelliJ IDEA 编辑视图:

messages是 Spring Boot 加载资源文件默认采用的名称(basename),如果你所使用的资源文件名称不是以messages命名或所使用的资源文件不是在src/main/resources根目录,你可以通过spring.messages.basename属性来做具体的配置。如,资源文件messages.propertiesmessages_en_US.properties假设它们所在的目录位置是src/main/resources/i18n

application.properties 配置示例:

1
spring.messages.basename:i18n/messages

静态文本消息示例:

1
2
<!-- 北京欢迎你! -->
<p th:text="#{welcom.message}"></p>

3.1.4 @{…} 链接表达式

链接表达式@{}是专门用来处理 URL 链接地址的

(1)绝对地址示例:

1
2
<!-- https://fanlychie.github.io -->
<p th:text="@{https://fanlychie.github.io}"></p>

(2)页面相对地址示例:

1
2
<!-- commons/base.html -->
<p th:text="@{commons/base.html}"></p>

(3)上下文相对地址(相对于当前的服务)示例:

1
2
<!-- /css/mian.css -->
<p th:text="@{/css/mian.css}"></p>

(4)服务器相对地址(相对于部署在同一个服务器中的不同服务)示例:

1
2
<!-- /image/upload -->
<p th:text="@{~/image/upload}"></p>

3.1.5 ~{…} 片段表达式

片段表达式~{}可以用来引用一段公共的 HTML 代码片段

语法 描述
~{templatename} 引用整个模板文件的代码片段
~{templatename :: selector} selector 可以是 th:fragment 指定的名称或其他选择器。如类选择器、ID选择器等
~{::selector} 相当于 ~{this :: selector},表示引用当前模板定义的代码片段

在 Thymeleaf 模板文件中,你可以使用th:fragment属性来定义一段公共的代码片段,然后你可以通过使用th:insertth:replaceth:include(Thymeleaf 3.0 开始不再推荐使用,本文也将不再介绍它)属性来将这些公共的代码片段引入到模板文件中来。

(1)src/main/resources/templates/base.html,通过th:fragment属性定义一段公共的代码片段:

1
<div id="footer" th:fragment="footerFragment">&copy; 2017 fanlychie</div>

(2)src/main/resources/templates/index.html,通过th:insert属性引用一段公共的代码片段:

1
<div th:insert="~{base :: footerFragment}"></div>

使用th:fragment属性定义代码片段时,你还可以声明一组参数:

1
2
3
4
5
6
7
8
9
<div th:fragment="crumbs(parent, child)">
<i th:text="${parent}"></i> <i th:text="${child}"></i>
</div>

<!--
<i>用户中心</i>
<i>我的订单</i>
-->
<div th:insert="::crumbs('用户中心', '我的订单')"></div>

此外,我们还可以通过类选择器、ID选择器等来引用公共的代码片段:

1
<div th:insert="~{base :: #footer}"></div>

3.1.6  内置对象

对象 描述
#ctx 上下文对象
#vars 同#ctx,表示上下文变量
#locale 上下文本地化(特定地理区域)变量,可参考java.utils.Locale
#request HttpServletRequest对象
#response HttpServletResponse 对象
#session HttpSession对象
#servletContext ServletContext 对象
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
<!-- #ctx  -->
<p th:text="${#ctx.getLocale()}"></p>
<p th:text="${#ctx.getVariable('message')}"></p>
<!-- #vars -->
<p th:text="${#vars.getLocale()}"></p>
<p th:text="${#vars.getVariable('message')}"></p>
<!-- #locale -->
<p th:text="${#locale}"></p>
<p th:text="${#locale.country}"></p>
<!-- #request -->
<p th:text="${#request.serverName}"></p>
<p th:text="${#request.serverPort}"></p>
<p th:text="${#request.requestURL}"></p>
<!-- #response -->
<p th:text="${#response.status}"></p>
<p th:text="${#response.characterEncoding}"></p>
<!-- #session -->
<p th:text="${#session.id}"></p>

3.1.7 工具类

对象 描述
#message 消息工具类,与 #{…} 作用相同
#uris 地址相关的工具类
#conversions 对象转换工具
#dates 日期转换工具
#calendars 日历工具类
#numbers 数字工具类
#strings 字符串工具类
#objects 对象工具类
#bools 布尔工具类
#arrays 数组工具类
#list List工具类
#set Set工具类
#maps Map工具类
1
2
<p th:text="${#strings.isEmpty(message)}"></p>
<p th:text="${#dates.format(now, 'yyyy-MM-dd HH:mm:ss')}"></p>

3.2 字面值

所谓字面值,首先它不是一个变量,它是一个具体的确切的值,通常这些值是比较简单的,例如:18'welcome'等,它们没有名称,以至于我们只能用值来称呼它们,因此我们称其为字面值。

3.2.1 文字字面值

文字字面值是用单引号引起来的任何字符内容,如果字符内容里面含有单引号,则需要进行转义:

1
2
3
4
<!-- Welcome to BeiJing! -->
<p th:text="'Welcome to BeiJing!'"></p>
<!-- 'Welcome to BeiJing!' -->
<p th:text="'\'Welcome to BeiJing!\''"></p>

3.2.2 数字字面量

1
2
3
4
<!-- 2017 -->
<p th:text="2017"></p>
<!-- 2018 -->
<p th:text="2017 + 1"></p>

3.2.3 布尔字面值

1
2
3
4
<!-- false -->
<p th:text="1 > 2"></p>
<!-- 否 -->
<p th:text="1 > 2 ? '是' : '否'"></p>

3.2.4 空字面值

1
2
<!-- false -->
<p th:text="${user == null}"></p>

3.2.5 字面令牌

字面令牌(Literal Tokens)的内容只能含有(不能含有空格、特殊符号等):

  • 大写或小写的字母、中文等不含空格和特殊符号的文本
  • 0 到 9 的数字
  • 中括号
  • 下划线
  • 连字符(-)
  • 点符号(.)

实际上,数字、布尔和空字面值都是字面令牌的特殊情况。字面令牌能够用来对标准表达式语法进行简化,我们可以将包裹它的内容的单引号去掉:

1
2
3
<p th:text="Welcome to BeiJing!"></p>
等效于
<p th:text="'Welcome to BeiJing!'"></p>

3.3 文本操作

我们可以对文本内容进行两种常用的操作,它们分别为字符串连接和字符串替换

3.3.1 字符串连接

1
2
<!-- Welcome to BeiJing! -->
<p th:text="'Welcome to ' + ${location} + '!'"></p>

3.3.2 字面值替换

符号||可以用来将字面值和表达式包裹起来,这样就能方便的替换变量的值,而不需要使用+连接符:

1
2
<!-- Welcome to BeiJing! -->
<p th:text="|Welcome to ${location}!|"></p>

3.4 算数运算

支持+(加)、-(减)、*(乘)、/(除)、%(模)运算:

1
2
3
4
5
6
<!-- 6 -->
<p th:text="4 + 2"></p>
<!-- 2 -->
<p th:text="${pagination.page + 1}"></p>
<!-- 2 -->
<p th:text="${pagination.page} + 1"></p>

3.5 布尔运算

支持and(且)、or(或)、!(非)、not(非)运算:

1
2
3
4
<p th:text="${user.online and user.vip}"></p>
<p th:text="${user.online or user.vip}"></p>
<p th:text="${!user.online}"></p>
<p th:text="${not user.online}"></p>

3.6 比较和相等

支持<lt)、>gt)、<=le)、>=ge)、==eq)、!=ne):

1
2
3
4
5
6
<p th:text="${user.age < 60}"></p>
<p th:text="${user.age <= 60}"></p>
<p th:text="${user.age > 18}"></p>
<p th:text="${user.age >= 18}"></p>
<p th:text="${user.age == 18}"></p>
<p th:text="${user.age != 18}"></p>

3.7 条件运算

三元运算符:(if) ? (then) : (else)

1
<p th:text="${user.online ? '在线' : '离线'}"></p>

二元运算符:(value) ?: (defaultValue)

其中,value非空(null)即真,条件为真时输出value,否则输出defaultValue

1
2
3
4
<!-- 你还没有登录,请先登录 -->
<p th:text="${token} ?: '你还没有登录,请先登录'"></p>
<!-- fanlychie@gmail.com -->
<p th:text="${user.email} ?: '你还没有绑定邮箱'"></p>

3.8 无操作符

当模板运行在服务器端时,Thymeleaf 会解析th:*属性的具体值替换标签体的内容。无操作符(_)则允许你使用原型标签体的内容作为默认值:

1
2
<!-- 你还没有登录,请先登录 -->
<p th:text="${token} ?: _">你还没有登录,请先登录</p>

4. 使用文本

首先介绍两个最基础的th:*th:textth:utext,它们都是用于处理文本消息内容

4.1 th:text

在标签体中展示表达式评估结果的文本内容:

1
<p th:text="${message}"></p>

使用外部化的文本内容:

1
<p th:text="${message}">Welcome to BeiJing!</p>

当它作为静态文件直接运行时,浏览器会自动忽略它不能识别的th:text属性,而显示<p>标签体的文本内容Welcome to BeiJing!

当它作为模板文件运行在服务器端时,th:text属性的具体值将会替换<p>标签体的文本内容。

4.2 th:utext

属性th:utextth:text的区别在于:

  • th:text默认会对含有 HTML 标签的内容进行字符转义;
  • th:utext(Unescaped Text)则不会对含有 HTML 标签的内容进行字符转义;

假设:message = "<b>Welcome to BeiJing!</b>"

(1)使用th:text属性:

1
<p th:text="${message}"></p>

th:text效果:Welcome to BeiJing!

(2)使用th:utext属性:

1
<p th:utext="${message}"></p>

th:utext效果:Welcome to BeiJing!

5. 设置属性值

在 Thymeleaf 模板文件中,你可以使用th:*(或者使用th:attr属性)来设置任意的 HTML5 标签属性的值。不仅如此,你还可以th:*-*来同时为多个不同的标签属性设置相同的一个值,甚至你可以使用th:attrappendth:attrprepend来追加新的值到现有的标签属性值中。

5.1 th:attr

这种方式是不被推荐的,了解一下就行。下面是用th:attr="href=..."来设置标签href属性的值:

1
<a th:attr="href=@{https://www.google.com.hk}">谷歌一下你就知道</a>

5.2 th:*

显然th:attr="href=@{http://www.baidu.com}"不够简洁,我们更推荐下面的这种语法:

1
<a th:href="@{https://www.google.com.hk}">谷歌一下你就知道</a>

其中th:*中的*可以是 HTML5 支持的任意属性名称,甚至这些属性名称可以是自定义的:

1
2
<!-- <div item-id="1001">Welcome to BeiJing!</div> -->
<div th:item-id="${user.id}">Welcome to BeiJing!</div>

5.3 th:-

如果想要同时为标签的多个不同属性设置相同的一个值,可以使用th:*-*的语法:

1
<img src="logo.png" th:alt-title="LOGO图片">

它相当于

1
<img src="logo.png" th:alt="LOGO图片" th:title="LOGO图片">

5.4 th:attrappend & th:attrprepend

th:attrappendth:attrprepend可以将表达式的结果分别追加到指定的属性值之后和之前。

1
2
3
4
<!-- <button class="btn enable">购买</button> -->
<button class="btn" th:attrappend="class=${outOfStock} ? ' enable' : ' disable'">购买</button>
<!-- <button class="enable btn">购买</button> -->
<button class="btn" th:attrprepend="class=${outOfStock} ? 'enable ' : 'disable '">购买</button>

另外,还有两个常用的具体附加属性th:classappend="..."th:styleappend=""

它们分别用来代替th:attrappend="class=..."th:attrappend="style=..."

1
2
<!-- <button class="btn enable">购买</button> -->
<button class="btn" th:classappend="${outOfStock} ? ' enable' : ' disable'">购买</button>

5.5 布尔属性

在 HTML 中有些属性是布尔属性,布尔属性是指没有值的属性,如readonlycheckedselected等。它们若存在那就意味着值为 true

1
2
3
4
5
6
7
8
9
10
<input type="checkbox" name="rememberme" checked /> 记住我
<input type="radio" name="sex" value="male" checked>
<input type="radio" name="sex" value="female">
<input type="text" name="appId" value="J123654" readonly>
<select>
<option selected>北京</option>
<option>上海</option>
<option>广州</option>
<option>深圳</option>
</select>

Thymeleaf 也允许我们通过th:*(这里的*表示任意的布尔属性) 来选择是否使用这些布尔属性。

1
<input type="checkbox" name="rememberme" ch:checked="${rememberme}" /> 记住我

如果表达式的结果为true,则自动勾选复选框,若为false,则不会自动勾选

6. 遍历

遍历(迭代)的语法th:each="自定义的元素变量名称 : ${集合变量名称}"

1
2
3
4
5
6
<div>
<spn>你所在城市:</spn>
<select name="mycity">
<option th:each="city : ${cities}" th:text="${city.name}"></option>
</select>
</div>

属性th:each提供了一个用于跟踪迭代的状态变量,它包含以下几个属性:

属性 类型 描述
index int 当前迭代的索引,从0开始
count int 当前迭代的计数,从1开始
size int 集合中元素的总个数
current int 当前的元素对象
even boolean 当前迭代的计数是否是偶数
odd boolean 当前迭代的计数是否是奇数
first boolean 当前元素是否是集合的第一个元素
last boolean 当前元素是否是集合的最后一个元素

状态变量的使用语法:th:each="自定义的元素变量名称, 自定义的状态变量名称 : ${集合变量名称}"

1
2
3
4
5
6
<div>
<spn>所在城市:</spn>
<select name="mycity">
<option th:each="city, status : ${cities}" th:text="${city.name}" th:item-index="${status.count}"></option>
</select>
</div>

不管什么时候,Thymeleaf 始终会为每个th:each创建一个状态变量,默认的状态变量名称就是自定义的元素变量名称后面加Stat字符串组成:

1
2
3
4
5
6
<div>
<spn>所在城市:</spn>
<select name="mycity">
<option th:each="city : ${cities}" th:text="${city.name}" th:item-index="${cityStat.count}"></option>
</select>
</div>

7. 条件判断

条件判断语句有三种,分别是:th:if,th:unless,th:swith

7.1 th:if

当表达式的评估结果为真时则显示内容,否则不显示

1
<a th:href="@{/user/order(uid=${user.id})}" th:if="${user != null}">我的订单</a>

真假评估的依据:

  • 当表达式的值不为空(null)时
    • 如果表达式的值是一个布尔类型,且值为true评估为真,否则为假
    • 如果表达式的值是一个数字类型,且值为非0评估为真,否则为假
    • 如果表达式的值是一个字符类型,且值为非0评估为真,否则为假
    • 如果表达式的值是一个字符串类型,且值为非"false""off""no"评估为真,否则为假
    • 如果表达式的值不是一个布尔数字字符字符串评估为真
  • 当表达式的值为空(null)时,评估结果为假

因此,上面代码我们也可以简写成:

1
<a th:href="@{/user/order(uid=${user.id})}" th:if="${user}">我的订单</a>

但是,为了代码的可读性,我们并不建议这样使用

7.2 th:unless

th:unlessth:if判断恰好相反,当表达式的评估结果为假时则显示内容,否则不显示:

1
<a th:href="@{/user/order(uid=${user.id})}" th:unless="${user == null}">我的订单</a>

7.3 th:swith

多路选择语句,它需要搭配th:case来使用:

1
2
3
4
<div th:switch="${user.role}">
<p th:case="admin">管理员</p>
<p th:case="user">普通用户</p>
</div>

8. 定义局部变量

使用th:with属性可以定义局部变量

1
2
3
<p th:with="name='fanlychie'">
<span th:text="${name}"></span>
</p>

同时定义多个局部变量时,用英文,号分隔开:

1
2
3
<p th:with="name=${user.name},age={user.age}">
......
</p>

9. 注释

9.1 标准注释

语法:<!-- ... --->,注释的代码块会在文件源代码中显示出来。

9.1.1 单行注释

1
<!-- <span>${message}</span> --->

9.1.2 多行注释

1
2
3
4
5
6
<!--
<div th:switch="${user.role}">
<p th:case="admin">管理员</p>
<p th:case="user">普通用户</p>
</div>
--->

9.2 解析器级注释

语法:<!--/* ... */-->,注释的代码块会在引擎解析的时候抹去

9.2.1 单行注释

1
<!--/* <span>${message}</span> */-->

9.2.2 多行注释

1
2
3
4
5
6
<!--/*-->
<div th:switch="${user.role}">
<p th:case="admin">管理员</p>
<p th:case="user">普通用户</p>
</div>
<!--*/-->

10. 内联表达式

内联表达式允许我们直接在 HTML 文本中使用标准表达式,而不需要使用th:*标签属性

10.1 [[…]]

[[]]相当于th:text,对含有HTML标签的内容自动进行字符转义

1
<p>The message is : [[${htmlContent}]]</p>

10.2 [(…)]

[()]相当于th:utext,对含有 HTML 标签的内容不进行字符转义

1
<p>The message is : [(${htmlContent})]</p>

10.3 th:inline

我们已经了解到,使用[[]][()]语法可以直接在 HTML 文本中使用标准表达式,如果想要使用更多高级的功能,需要使用th:inline属性来激活,它的取值如下:

描述
none 禁止内联表达式,可以原样输出[[]]和[()]字符串
text 文本内联,可以使用th:each等高级语法
css 样式内联,如:<style th:inline="css">
javascript 脚本内联,如:<style th:inline="javascript">

10.3.1 none

1
2
<!-- [[1, 2], [3, 4]] -->
<p th:inline="none">[[1, 2], [3, 4]]</p>

10.3.2 text

1
2
3
4
5
6
<!-- 北京 上海 广州 深圳 -->
<p th:inline="text">
[# th:each="city : ${cities}"]
[(${city.name})]
[/]
</p>

10.3.3 css

1
2
3
4
5
<style th:inline="css">
body {
background-color:[[${bgColor}]];
}
</style>

10.3.4 javascript

1
2
3
4
<script th:inline="javascript">
var user = [[${user}]];
alert("用户名:" + user.name);
</script>