侧边栏壁纸
博主头像
张种恩的技术小栈博主等级

行动起来,活在当下

  • 累计撰写 748 篇文章
  • 累计创建 65 个标签
  • 累计收到 39 条评论

目 录CONTENT

文章目录

SpringBoot(13)之Thymeleaf使用

zze
zze
2018-03-01 / 0 评论 / 0 点赞 / 728 阅读 / 29815 字

Thymeleaf 介绍

Thymeleaf 是一种用于 Web 和独立环境的现代服务器端的 Java 模板引擎。

Thymeleaf 的主要目标是将优雅的自然模板带到开发工作流程中,并将 HTML 在浏览器中正确显示,并且可以作为静态原型,让开发团队能更容易地协作。Thymeleaf 能够处理 HTML,XML,JavaScript,CSS 甚至纯文本。

Thymeleaf 使用 Spring 框架的模块,与许多常见的工具集成在一起,并且可以插入自己的功能,是现代 HTML5 JVM Web 开发的理想选择,尽管 Thymeleaf 还有更多其它的功能。

Thymeleaf 建立在自然模板的概念之上,以不影响模板作为设计原型的方式将其逻辑注入到模板文件中。 这改善了设计沟通,弥合了前端设计和开发人员之间的理解偏差。

关注文章首部微信公众号发送 #97_thymeleaf_doc 获取 Thymeleaf 中文离线文档 | Thymeleaf 官网

SpringBoot 中使用 Thymeleaf

准备

引入

Thymeleaf 是 SpringBoot 推荐使用的一款模板引擎框架,要引入很简单,SpringBoot 为它提供了场景启动器:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-thymeleaf</artifactId>
</dependency>

切换版本

如 SpringBoot 1.5.19 版本使用的 Thymeleaf 版本默认为 2.1.6,如果想切换到 3.0 以上,直接覆盖它的版本定义属性即可,要注意的是需要同时更新它的布局功能支持程序的版本:

<thymeleaf.version>3.0.2.RELEASE</thymeleaf.version>
<!--布局功能支持程序,thymeleaf 使用 3.0 版本以上时支持程序要使用 2.0 以上-->
<thymeleaf-layout-dialect.version>2.1.1</thymeleaf-layout-dialect.version>

入门程序

分析

可以先看下 SpringBoot 中 Thymeleaf 的自动配置类:

@Configuration
@EnableConfigurationProperties(ThymeleafProperties.class)
@ConditionalOnClass(SpringTemplateEngine.class)
@AutoConfigureAfter(WebMvcAutoConfiguration.class)public class ThymeleafAutoConfiguration {

查看它的属性配置类:

@ConfigurationProperties(prefix = "spring.thymeleaf")
public class ThymeleafProperties {
    private static final Charset DEFAULT_ENCODING = Charset.forName("UTF-8");
    private static final MimeType DEFAULT_CONTENT_TYPE = MimeType.valueOf("text/html");
    public static final String DEFAULT_PREFIX = "classpath:/templates/";
    public static final String DEFAULT_SUFFIX = ".html";

一目了然,Thymeleaf 默认使用的模板路径为 classpath:/templates/ ,且可省略后缀 .html ,下面我们就开始在 SpringBoot 项目中使用 Thymeleaf。

步骤

1、创建测试控制器:

import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;

@Controller
public class TestController {

    @RequestMapping("test")
    public String test(Model model){
        // 传值
        model.addAttribute("name", "张三");
        // SpringBoot 会找到 classpath:templates/test.html 使用 thymeleaf 渲染
        return "test";
    }
}

2、新建模板页面:

<!-- templates/test.html -->
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>test</title>
</head>
<body>
<h1 th:text="${name}"></h1>
</body>
</html>

3、测试,启动项目,访问 localhost:8080/test

image.png

表达式

变量表达式

Thymeleaf 的变量表达式类似于 EL 表达式,通过 ${} 取值。

// controller
List<String> nameList = new ArrayList<>();
nameList.add("张三");
nameList.add("李四");
nameList.add("王五");

model.addAttribute("name", "张三");
model.addAttribute("nameList", nameList);
<!-- html -->
<!--取值-->
<span th:text="${name}"></span>
<hr>
<!--循环-->
<ul>
    <li th:each="name : ${nameList}"><span th:text="${name}"/></li>
</ul>

image.png

选择变量表达式

与变量表达式很像,不过它需要预先选择一个对象作为上下文变量容器。

// bean
public class User {
    public User(){}
    public User(String name, Integer age) {
        this.name = name;
        this.age = age;
    }

    private String name;

    private Integer age;

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public Integer getAge() {
        return age;
    }

    public void setAge(Integer age) {
        this.age = age;
    }
}
// controller
model.addAttribute("user", new User("张三", 18));
<!-- html -->
<div th:object="${user}">
    <p>姓名:<span th:text="*{name}"></span></p>
    <p>年龄:<span th:text="*{age}"></span></p>
</div>

URL 表达式

URL 表达式可以帮助我们更轻松的动态拼装请求 URL。

<!--() 中可以指定要传递的参数-->
<span th:text="@{/order/details(type=1,keyword=ff)}"></span>

表达式支持语法

字面量

文本文字 : 'one text', 'Another one!',…
数字文本 : 0, 34, 3.0, 12.3,…
布尔文本 : true, false
空 : null
文字标记 : one, sometext, main,…

文本操作

字符串连接 : +
文本替换 : |The name is ${name}|

算术运算

二元运算符 : +, -, *, /, %
减号(单目运算符) : -

布尔操作

二元运算符 : and, or
布尔否定(一元运算符) : !, not

比较

比较 : >, <, >=, <= (gt, lt, ge, le)
等值运算符 :==, != (eq, ne)

条件运算

If-then : (if) ? (then)  # 例:<span th:text="${name} == '张三' ? 'Administrator'"/>
If-then-else : (if) ? (then) : (else)  # 例:<span th:text="${name} == '张三' ? 'Administrator' : (${name} ?: 'Unknown')"/>
Default : (value) ?: (defaultvalue)  # 例:<span th:text="${name} ?: 'Unknown'"/>

常用标签属性

关键字功能介绍案例
th:id替换id <input th:id="'xxx' + ${collect.id}"/> 
th:text文本替换 <p th:text="${collect.description}">description</p> 
th:utext支持html的文本替换 <p th:utext="${htmlcontent}">conten</p> 
th:object替换对象 <div th:object="${session.user}"> 
th:value属性赋值 <input th:value="${user.name}" /> 
th:with变量赋值运算 <div th:with="isEven=${prodStat.count}%2==0"></div> 
th:style设置样式 <span th:style="'display:' + @{(${sitrue} ? 'none' : 'inline-block')} + ''"/>  
th:onclick点击事件 <button th:onclick="'getCollect()'"></button> 
th:each属性赋值 <tr th:each="user,userStat:${users}"></tr>  
th:if判断条件 <a th:if="${userId == collect.userId}" > 
th:unless和th:if判断相反
<a th:href="@{/login}" th:unless=${session.user != null}>Login</a>
th:href链接地址
<a th:href="@{/login}" th:unless=${session.user != null}>Login</a>
th:switch多路选择 配合th:case 使用 <div th:switch="${user.role}"> 
th:caseth:switch的一个分支 <p th:case="'admin'">User is an administrator</p> 
th:fragment布局标签,定义一个代码片段,方便其它地方引用 <div th:fragment="alert"> 
th:include布局标签,替换内容到引入的文件 <head th:include="layout :: htmlhead" th:with="title='xx'"></head> /> 
th:replace布局标签,替换整个标签到引入的文件 <div th:replace="fragments/header :: title"></div> 
th:selectedselected选择框 选中 <option th:selected="(${xxx.id} == ${configObj.dd})"></option> 
th:src图片类地址引入 <img class="img-responsive" alt="App Logo" th:src="@{/img/logo.png}" /> 
th:inline定义js脚本可以使用变量 <script type="text/javascript" th:inline="javascript"> 
th:action表单提交的地址 <form action="subscribe.html" th:action="@{/subscribe}"> 
th:remove删除某个属性

 <tr th:remove="all">  

1.all:删除包含标签和所有的孩子。2.body:不包含标记删除,但删除其所有的孩子。3.tag:包含标记的删除,但不删除它的孩子。4.all-but-first:删除所有包含标签的孩子,除了第一个。5.none:什么也不做。这个值是有用的动态评估。

th:attr设置标签属性,多个属性可以用逗号分隔

 <img th:attr="src=@{/image/aa.jpg},title=#{logo}"/> 此标签不太优雅,一般用的比较少。

一个标签内可以包含多个 th:x属性,其生效优先级顺序如下:

include、each、if/unless/switch/case、with、attr、attrprepend、attrappend、value、href、src、etc、text、utext、fragment、remove

常用操作

字符串拼接

<!--使用 + 号-->
<span th:text="'Welcome to our application, ' + ${name} + '!'"/> <br>
<!--使用 | 进行字符串格式化-->
<span th:text="|Welcome to our application, ${name}!|"/>

条件判断

<span th:if="${name}=='张三'">是张三</span>
<span th:unless="${name}=='张三'">不是张三</span>
<span th:text="${name} ?: 'Unknown'"/>
<span th:text="${name} == '张三' ? 'Administrator'"/>
<span th:text="${name} == '张三' ? 'Administrator' : (${name} ?: 'Unknown')"/>
<div th:switch="${name}">
    <span th:case="张三">name 为张三</span>
    <span th:case="李四">name 为李四</span>
</div>

循环

<ul>
    <li th:each="name,iterStat : ${nameList}" th:text="${iterStat.count} + ':'+ ${name}"></li>
</ul>
<!--
iterStat称作状态变量,属性有:
    index:当前迭代对象的index(从0开始计算)
    count: 当前迭代对象的index(从1开始计算)
    size:被迭代对象的大小
    current:当前迭代变量
    even/odd:布尔值,当前循环是否是偶数/奇数(从0开始计算)
    first:布尔值,当前循环是否是第一个
    last:布尔值,当前循环是否是最后一个
-->

组装 URL

<!--() 中可以指定要传递的参数-->
<form th:action="@{/order/details(type=1,keyword=ff)}" ></form>
<!--上述对应的 URL 为 /order/details?type=1&keyword=ff-->

常用内置对象

Thymeleaf 为我们提供了很多内置对象,通过 ${#内置对象名称} 即可访问到,下面列出一些比较常用的:

内置对象作用示例
dates日期操作
<span th:text="${#dates.format(currentDate,'yyyy-MM-dd HH:mm:ss')}"/><!--格式化日期-->
numbers数字格式化
<span th:text="${#numbers.formatDecimal(13.213, 0, 2)}"></span><!--此示例表示保留两位小数位,整数位自动  结果 13.21--><span th:text="${#numbers.formatDecimal(13.213, 3, 2)}"></span><!--此示例表示保留两位小数位,3位整数位(不够的前加0) 结果 013.21-->
lists列表操作
<p th:text="${#lists.size(nameList)}"/><!--获取列表长度-->
calendars日历操作
<p th:text="${#calendars.format(#calendars.createNow(),'yyyy-MM-dd HH:mm:ss')}"></p><!--格式化日期,与 #dates 相似-->
strings字符串操作 
<p th:text="${#strings.startsWith('abcde','aab')}"/><!--判断字符串是否以指定字符串开头-->
objects对象操作
<p th:text="${#objects.nullSafe(name,'Unknown')}"></p><!--判断指定对象是否为空,如果是空则返回指定默认值,否则原样返回-->
bools布尔值操作
<p th:text="${#bools.isFalse(1>2)}">aa</p><!--判断一个表达式结果是否为假-->
arrays数组操作
<p th:text="${#arrays.isEmpty(testArr)}"></p><!--判断一个数组是否为空-->
sets集合操作
<p th:text="${#sets.size(set)}"></p><!--获取一个集合中元素个数-->
maps地图操作
<p th:text="${#maps.containsKey(map,'key1')}"></p><!--判断一个 Map 中是否存在指定 key-->
aggregates统计运算
<p th:text="${#aggregates.avg(numArr)}"></p><!--计算一个数组中的平均值-->
messages属性文件取值 
<p th:text="${#messages.msg('hahah')}"/><!--取一个属性文件中的属性值,相当于 <p th:text="#{hahah}"/>-->
convertions类型转换
<p th:text="${#conversions.convert('213','java.lang.Integer')+23}"></p><!--将一个字符串转成 Integer 类型-->
execInfo模板信息
<p th:text="${#execInfo.getTemplateName()}"></p><!--获取运行时当前模板名称-->
request请求对象
<p th:text="${#request.method}"></p><!--通过请求对象获取当前的请求方法 #httpServletRequest 与之相同-->
response响应对象
<p th:text="${#response.getWriter().write('aaa')}"/><!--通过相应对象输出字符串 -->
session会话对象
<p th:text="${#session.getId()}"></p><!--通过会话对象获取当前会话id #httpSession 与之相同-->

布局功能

概述

在 Web 开发中,我们经常会将公共头,公共尾,菜单等部分提取成模板供其它页面使用。在 Thymeleaf 中,通过 th:fragmentth:includeth:replace、参数化模板配置、css 选择器加载代码块等实现。

Spring Boot 2.0 将布局单独提取了出来,需要单独引入依赖:thymeleaf-layout-dialect

<dependency>
    <groupId>nz.net.ultraq.thymeleaf</groupId>
    <artifactId>thymeleaf-layout-dialect</artifactId>
</dependency>

选择器使用

1、定义模板:

<!-- templates/common/header.html -->
<div class="header">
    这是头部
</div>
<!-- templates/common/body.html -->
<div class="body">
    这是主体
</div>
<!-- templates/common/fotter.html -->
<div class="footer">
    这是尾部
</div>

2、引用模板:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>模板测试</title>
</head>
<body>
    <!--insert 会将所有选择的标签及内容插入到当前标签内-->
    <div th:insert="common/header :: .header"></div>
    <!--replace 会让选择的标签替换当前的标签-->
    <div th:replace="common/body :: .body"></div>
    <!--include 会将选择的标签内容插入到当前标签内-->
    <div th:include="common/footer :: .footer"></div>
</body>
</html>

image.png

fragment 使用

1、定义模板块:

<!-- templates/common/fragment.html -->
<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>fragment Test</title>
</head>
<body>
<!--fragment 定义用于被加载的块-->
<span th:fragment="copy">msg from fragment</span>
<!--定义能接收参数的块-->
<span th:fragment="sayHello(msg, name)">[[|${msg} ${name}|]]</span>
</body>
</html>

2、使用模板块:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>模板测试</title>
</head>
<body>
<div th:include="common/fragment::copy"></div>
<div th:include="common/fragment::sayHello('hello','bob')"></div>
</body>
</html>

image.png

其中 th:includeth:insertth:replace 中的参数格式为 templatename::[domselector] ,其中 templatename 是模板名(如 footer ), domselector 是可选的 dom 选择器。如果只写 templatename ,不写 domselector ,则会加载整个模板。

0

评论区