Thymeleaf 入门基础

一,Thymeleaf 介绍

Thymeleaf 是一个 服务器 端 Java 模板引擎 , 类似于 FreeMarker 、 Velocity 等模板 。 它是
Java 服务器端 HTML5 开发的理想选择 。 并且能够很好的和 Spring Boot 集成 。

Thymeleaf 的主要目标是为 程序员 的开发工作流程带来优雅的自然模板 , 它可以作为静态原型来使用 ,
这对前后端分离的开发团队来说无疑是一种很好的体验 。

Thymeleaf 开箱即用 , 它提供了一套标准方言 , 所有的数据都是通过标签 th:* 来完成 。 如:

Welcome to BeiJing! 当它作为静态文件运行时 ,
Thymeleaf 会过虑 th: 标签属性 , 显示静态文本值 Welcome to BeiJing! 。

当它运行在服务器环境时 , Thymeleaf 会使用 th: 标签属性的值替换掉静态文本的值 , 动态显示 ${message}
的具体内容 。

除了上面提到的 th:text , Thymeleaf 还提供了许多其他功能 , 这些功能对于大多数情况是足够满足日常开发的需求的 ,
下面来一一做介绍 。

如何用Thymeleaf来创建一个有效的html页面呢?

1
2
3
4
5
6
7
8
9
10
11
<!DOCTYPE HTML>
<html xmlns:th="http://www.thymeleaf.org">
 <head>
 <title>Welcome Page</title>
 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
 </head>
 <body>
 <span th:text="${message}">Welcome to BeiJing!</span>
 </body>
 </body>
</html>

正如你所见,这是一个html页面文件,它可以在任何浏览器上正确的显示(浏览器会忽略它不能识别的标签 th:text),但这不是一个标准的html页面文件,因为html页面规范是不允许出现 th: 这些非标准属性的,而且我们还加了 xmlns:th=“http://www.thymeleaf.org” 属性。
实际上,我们不写 xmlns:th=“http://www.thymeleaf.org” 属性对页面是不产生什么影响的,它只是防止IDEA找不到像 th:属性命名空间的错误,但对页面的正常显示是不产生影响的。
因此,我们甚至可以不用 th:* 属性,我们可以切换到Thymeleaf的属性语法,用data-th 来代替 th:

1
2
3
4
5
6
7
8
9
10
11
<!DOCTYPE HTML>
<html>
 <head>
 <title>Welcome Page</title>
 <meta http-equiv="Content-Type" content="text/html; charset=UTF-8" />
 </head>
 <body>
 <span data-th-text="${message}">Welcome to BeiJing!</span>
 </body>
 </body>
</html>

html 5 规范是允许 data- * 这样自定属性的。至此,我们已经把模板改造成了一个真正有效的 HTML 文档。
th:* data-th-* 这两个符号是完全等效且是可以互换的 。 为了简单直观和代码的紧凑性 , 这里采用th:* 的表示方式 。


二,标准表达式语法

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

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

1:简单表达式
Thymeleaf 提供的简单表达式语法有 5 类 , 它们分别是:

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

${…}

下面这段代码是取出上下文环境中message变量的值。

1
<span th:text='${message}'></span>

内置对象
跟 JSP 相像的 , Thymeleaf 也为我们提供了一些内置的对象变量 , 它们都是以 # 开头表示:

语法 描述
#ctx 上下文对象
#vars 上下文变量
#locale 上下文区域变量
#request HttpServletRequest 对象
#response HttpServletResponse 对象
#session HttpSession 对象
#servletContext ServletContext 对象

举个例子:

1
2
3
4
<!-- zh_CN -->
<p th:text="${#locale}"></p>
<!-- name -->
<p th:text="${#session.user.name}"></p>


工具类
另外 , Thymeleaf 还提供了许多内置的 工具 类 , 它们也是以 # 开头:

对象 描述
#messages 消息工具类 , 与 #{} 作用相同;
#uris 地址相关工具类
#conversions 对象转换工具类
#dates 日期转换工具类
#calendars 日历工具类
#numbers 数字工具类
#strings 字符串工具类
#objects 对象工具类
#bools 布偶工具类
#arrays 数组工具类
#lists List工具类
#sets Sets工具类
#maps maps工具类

举个例子:

1
2
3
4
5
6
<p th:text="${#lists.isEmpty(cities)}"></p>
<p th:text="${not #lists.isEmpty(cities)}"></p>
<!-- 2020年4月4日 下午016时01分12秒  历史难忘的日子-->
<p th:text="${#dates.format(now)}"></p>
<!-- 2020年4月4日 下午016时01分12秒-->
<p th:text="${#dates.format(now, 'yyyy-MM-dd HH:mm:ss')}"></p>

*{…}

使用 * { } 表达式时,如果没有选择对象,那么, * { } ${} 表达式表达的效果是一样的。

1
2
<p th:text='*{user.name}'></p>
<p th:text='${user.name}'></P>

* { } 表达式的上下文是父标签( th:object )所选择的对象 , 而 ${} 表达式是面向整个上下文的 , 这是 * { } ${} 表达式语法的一个重要的区别 。

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

它相当于:

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

# {…}

首先我们了解一下 i18n 资源文件的命名规则:

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

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

处理静态消息

我们来创建以下几个文件:
1. /src/main/resources/i18n/messages.properties

welcome.message=北京欢迎你!

  1. /src/main/resources/i18n/messages_en_US.properties

welcome.message=Welcome to BeiJing!

  1. /src/main/resources/application.properties (Spring Boot)

spring.messages.basename:i18n/messages

举个例子:

1
<p th:text='#{welcome.message}'></p>

处理非静态消息

#{ } 表达式是不能直接处理非静态消息的,我们可以在资源文件中使用 占位符 { } 来处理非静态消息。

/src/main/resources/i18n/messages.properties

welcome.user={0},欢迎来到北京!

使用时,通过消息的键值以参数的形式来传递变量。

1
<p th:text='#{welcome.user(${user.name})}'></p>

@{…}

在WEB应用开发里面URL的链接地址是非常常见的,@{ } 表达式就是专门用来处理URL链接的:

1
2
3
4
5
6
7
8
9
10
<!-- /css/mian.css -->
<p th:text="@{/css/mian.css}"></p>
<!-- /css/mian.css?v=1.0 -->
<p th:text="@{/css/mian.css(v=1.0)}"></p>
<!-- /user/order?uid=1001 -->
<p th:text="@{/user/order(uid=${user.id})}"></p>
<!-- /user/order?uid=1001&status=PAIED -->
<p th:text="@{/user/order(uid=${user.id},status='PAIED')}"></p>
<!-- /user/1001/info -->
<p th:text="@{/user/{uid}/info(uid=${user.id})}"></p>

绝对地址:

1
2
<!-- https://blog.csdn.net/qq_44538738 -->
<a th:href='@{https://blog.csdn.net/qq_44538738}'></a>

页面相对地址(相对于当前服务):

1
2
<!-- commons/util.js -->
<script th:src="@{commons/util.js}"></script>

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

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

~{…}

当需要引用某个模板文件的公共的 HTML 代码片段时 , 可以使用 ~{ } 语法:

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

具体参考模板布局章节。


字面值

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


文字字面值

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

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

数字字面值

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

布尔字面值

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

空字面值

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

字面令牌

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

  • 大小写字母
  • 0 - 9 的数字
  • 中括号
  • 下划线
  • 连字符(-)
  • 点(.)
  • 中文(官方文档未提及)
  • 其它

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

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

文本操作

文本操作包含字符连接和替换的操作 。


字符串连接

不管是字面值还是表达式的结果 , 我们都可以使用 + 符号将它们连接起来:

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

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

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

算术运算

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

1
2
3
4
5
6
7
8
9
10
<!-- 6 -->
<p th:text="4 + 2"></p>
<!-- 2 -->
<p th:text="4 - 2"></p>
<!-- 8 -->
<p th:text="4 * 2"></p>
<!-- 2 -->
<p th:text="4 / 2"></p>
<!-- 0 -->
<p th:text="4 % 2"></p>

布偶运算
支持 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>

比较和相等

支持 < 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>

条件运算

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

1
2
<p th:text="${user.online ? '在线' : '离线'}"></p>
<p th:text="${user.online ? (user.vip ? 'VIP用户在线' : '普通用户在线') : '离线'}"></p>

二元运算符:(value) ?: (defaultValue) , 其中 , value 非空(null)即真 , 条件为真时输出 value , 否则输出 defaultValue

假设 token = null user.email = [email protected]

1
2
3
4
<!-- 你还没有登录<span class="fuhao">,</span>请先登录 -->
<p th:text="${token} ?: '你还没有登录<span class="fuhao">,</span>请先登录'"></p>
<!-- [email protected] -->
<p th:text="${user.email} ?: '你还没有绑定邮箱'"></p>

无操作符:

当模板运行在服务器端时 , 无论如何 Thymeleaf 都会用 th:* 属性的具体值替换标签体的内容 。

无操作符( _ )则允许使用原型标签体的内容作为默认值:

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

使用文本

了解了标准表达式的语法之后 , 接下来介绍如何正确的使用它们 。

标准表达式通常需要搭配 th:* 属性来使用 , 这样 Thymeleaf 引擎才能正确的解析出表达式的结果 。

首先介绍两个最基础的 th:* 属性 , 它们分别是: th:text th:utext , 这两个属性都是用于处理文本内容 。


th:text

在标签中显示表达式的值的文本内容:

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

使用外部化文本内容:

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

当它作为静态文件直接允许时,浏览器会自动忽略不能识别的 th:text 标签,而显示 span 标签体的文本内容。

当它作为模板文件直接运行在服务器时, th:text 属性标签的值 会直接替换 span 里的文本内容。


th:utext

th:text th:utext 的区别在于:

  • th:text 默认会对含有HTML 标签的内容进行转义。
  • th:utext 则不会对含有HTML 标签的内容进行转义。

假设:message = “Welcome to BeiJing!

th:text 例子:

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

th:text 结果:

1
<b>Welcome to BeiJing!</b>

th:utext 例子:

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

th:utext 结果:

Welcome to BeiJing!


设置属性值

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


th:arr

这种方式是不推荐的 , 了解一下就好 。 下面是用 th:attr=“href=…” 来设置标签 href 属性的值:

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

th:*
上面 th:attr="href=@{http://www.baidu.com}" 语法显得臃肿甚至不简洁 , 我们更推荐下面的这种语法:

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

这样代码就简洁多了 , 其中 th:* 中的 * 可以是 HTML5 支持的任意属性名称 , 甚至这些属性名称可以是自定义的:

1
2
3
4
<!-- <div item-id="1001">Welcome to BeiJing!</div> -->
<div th:item-id="${user.id}">Welcome to BeiJing!</div>
<div th:class='box'></div>
<input th:value='123' />

th: * - *

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

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

它相当于:

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

th:attrappend & th:attrprepend

th:attrappend th: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>

另外 , Thymeleaf 还提供了两个常用的具体附加属性 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>

布尔属性

在 HTML 中有些属性是布尔属性 , 布尔属性是指没有值的属性 , 如 readonly checked selected 等 。 它们若存在那就意味着值为 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 , 则不会自动勾选 。


遍历

遍历(迭代)的语法 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>

条件判断

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


th:if

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

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

真假评估的依据:

  • 当表达式的值不为 null 时
    1. 如果表达式的值是一个布尔类型 , 且值为 true 评估为真 , 否则为假;
    2. 如果表达式的值是一个数字类型 , 且值为非 0 评估为真 , 否则为假;
    3. 如果表达式的值是一个字符类型 , 且值为非 0 评估为真 , 否则为假;
    4. 如果表达式的值是一个字符串类型 , 且值为非 “false” 、 “off” 、 “no” 评估为真 , 否则为假;
    5. 如果表达式的值是一个字符串类型 , 且值为非 “false” 、 “off” 、 “no” 评估为真 , 否则为假;
  • 当表达式的值为 null 时,评估为假。

因此上面的代码我们简写成:

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

但是 , 为了代码的可读性 , 我们并不建议这样使用(一眼比较难看出,容易误导新手) 。


th:unless

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

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

th:swith

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

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

模板布局

在 WEB 应用开发中 , 我们通常会将一些公共的代码片段如页眉 、 页脚 、 菜单等抽取出来放到一个或多个独立的文件里面 , 其他模板文件可以通过标签轻松的将这些代码片段引用进来 , 从而实现代码的复用性 。

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


th:fragment

我们可以通过 th:fragment 来定义一段公用的代码片段:

templates/footer.html

1
<div th:fragment="base">? 2017 fanlychie</div>

th:insert

templates/welcome.html 文件中引用footer.html base 代码片段的方式为:

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

其中 , ~{ } 是可选的 , 我们可以去掉这层的包裹:

1
<div th:insert="footer :: base"></div>

如果 footer.html welcome.html 不在同级目录 , 如 templates/commons/footer.html

1
<div th:insert="@{commons/footer} :: base"></div>

类选择器 、 ID选择器等

除了可以使用 th:fragment 来定义一段公用的代码片段外 , 我们也可以使用类选择器 、 ID选择器等:

templates/footer.html

1
<div id="base">? 2017 fanlychie</div>

templates/welcome.html 文件中引用 footer.html id=“base” 片段的方式为:

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

th:replace

th:insert 相似的 ,th:replace 也可以用来引用公共的代码片段 。
不同的是 , th:insert 是直接将代码片段插入到标签体内 , 而 th:replace 则是用代码片段直接替换标签体内容 。

举个例子:

1
2
3
4
<!-- <div><div id="base">? 2017 fanlychie</div></div> -->
<div th:insert="footer :: base"></div>
<!-- <div id="base">? 2017 fanlychie</div> -->
<div th:replace="footer :: base"></div>

th:fragment 参数使用

使用 th:fragment 来定义一个代码片段时 , 它允许你指定一组参数 , 语法:

1
2
3
4
5
6
<font color='red'>th:fragment="fragmentname(arg1, arg2, ...)"</font>
<div th:fragment="crumbs(parent, child)">
 <span th:text="|${parent} > ${child}|"></span>
</div>
<!-- 用户中心 > 我的订单 -->
<div th:replace="::crumbs('用户中心', '我的订单')"></div>

th:assert

断言 , 它可以同时指定多个表达式 , 每个表达式的结果评估值必须为 true , 否则它将会抛出一个异常 。 语法:th:assert=“expression1, expression2…”

1
<div th:replace="::crumbs(${parentMenuName}, ${childMenuName})" th:assert="${!#strings.isEmpty(parentMenuName), !#strings.isEmpty(childMenuName)}"></div>

定义局部变量

使用 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>

注释

下面介绍常见两种注释:

标准注释

语法: , 注释的代码块会在引擎解析的时候抹去 。

单行注释:

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

多行注释:

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

内联表达式

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

[[…]]

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

htmlContent = “Welcome to BeiJing!

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

输出:

1
The message is : <b>Welcome to BeiJing!</b>

[(…)]

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

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

输出:

The message is : Welcome to BeiJing!


th:inline

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

描述
none 禁止使用内联语法,会直接输出 [[ ]] and[( )]
text 文本内联 , 可以使用 th:each 等高级语法
css 样式内联 , 如: <`style th:inline="css"`>
javascript 脚本内联 , 如: <`style th:inline="javascript"`>

none

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

text

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

css

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

javascript

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

原文链接:

https://www.codercto.com/a/9497.html