Spring Boot 模板引擎Thymeleaf集成

版权声明:本文为博主原创文章,转载请附上博文链接! https://blog.csdn.net/Little_fxc/article/details/83302884

Spring Boot 模板引擎Thymeleaf集成

一、Thymeleaf介绍

Thymeleaf是一种Java XML / XHTML / HTML5模板引擎,可以在Web和非Web环境中使用。它更适合在基于MVC的Web应用程序的视图层提供XHTML / HTML5,但即使在脱机环境中,它也可以处理任何XML文件。它提供了完整的Spring Framework集成。

二、Thymeleaf基础使用

Thymeleaf的使用是由两部分组成的:标签 + 表达式,标签是Thymeleaf的语法结构,而表达式就是语法里的内容实现。

通过标签 + 表达式,让数据和模板结合,最终转换成html代码,返回给用户。

Thymeleaf基础使用分为2部分:

  1. 标签使用
  2. 表达式使用

1.标签使用

1.1 th:text 基础信息输出

HTML代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<span th:text="${name}"></span>
</body>
</html>

 

Java代码:

package com.littlefxc.blog.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

/**
 * @author fengxuechao
 */
@Controller
public class IndexController {


    @RequestMapping("/")
    public ModelAndView index(ModelAndView modelAndView) {
        modelAndView.setViewName("index");
        modelAndView.addObject("name", "我是冯雪超");
        return modelAndView;
    }
}

最终效果:

1.2 th:utext html内容输出

使用"th:text"是对内容的原样输出,使用“th:utext”可以进行html标签输出。

Java代码:

package com.littlefxc.blog.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

/**
 * @author fengxuechao
 */
@Controller
public class IndexController {


    @RequestMapping("/")
    public ModelAndView index(ModelAndView modelAndView) {
        modelAndView.setViewName("index");
        modelAndView.addObject("name", "<span style='color:red'>我是冯雪超</span>");
        return modelAndView;
    }
}

 

HTML代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<p th:utext="${name}"></p>
</body>
</html>

 

 

展示效果:

1.3 th:if, th:unless 条件判断

 

<span th:if="${age > 18}">

    成年

</span>

<span th:unless="${age > 18}">

    未成年

</span>

th:if为满足条件的业务处理,th:unless正好相反,是除去的意思。

1.4 th:switch, th:case 多条件判断

<div th:switch="${age}">

    <span th:case="18">18岁</span>

    <span th:case="19">19岁</span>

    <spa th:case="*">其他</spa>

</div>

注意: 默认选项使用th:case="*" 指定。

1.5 th:each 循环

HTML代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:each="name,item:${names}">
    <span th:text="${item.count}"></span>
    <span th:text="${name}"></span>
</div>
</body>
</html>

 

 

Java代码:

package com.littlefxc.blog.controller;

import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

import java.util.Arrays;

/**
 * @author fengxuechao
 */
@Controller
public class IndexController {


    @RequestMapping("/")
    public ModelAndView index(ModelAndView modelAndView) {
        modelAndView.setViewName("index");
        modelAndView.addObject("names", Arrays.asList("小王", "小明", "小红"));
        return modelAndView;
    }
}

 

 

访问效果如下:

其中item为每行的详细值,常用key值如下:

  • index 下标,从0开始
  • count 第x个,从1开始
  • size 这个集合的大小
  • current 当前行的值

1.6 th:fragment、th:insert、th:replace、th:include 代码片段复用

  • th:fragment标签是声明代码片段,用于解决代码复用的问题,好比Java程序写的公用代码一样,每个需要的地方都可以直接调用;
  • th:insert 引用fragment的代码,保留自己的主标签;
  • th:replace 引用fragment的代码,不保留自己的主标签;
  • th:include 使用类似th:replace,Thymeleaf3.0之后不推荐使用;

footer.html页面代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div th:fragment="copyright">
    © 著作权归 冯雪超 所有
</div>

<div th:fragment="about">
    关于
</div>

<div th:fragment="links">
    CCTV
</div>
</body>
</html>

 

声明了3个代码片段,copyright、about和links。

Index.html页面代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:replace="footer :: copyright"></div>

<div th:insert="footer :: about"></div>

<div th:include="footer :: links"></div>
</body>
</html>

 

其中第一个div引用了footer.html 的 copyright 代码片段,第二个div引用了 footer.html 的 about 代码片段。

双冒号的理解: 其中使用“::”双冒号来完成对页面片段的引用,有点像php里面的语法,使用双冒号来表示对类的静态属性和方法进行直接引用。

执行效果如下图:

总结: 可以很清晰的看出th:insert、th:replace、th:include之间的区别,在于是否保留自己的主标签,th:include 在3.0之后已经不推荐使用了,可以使用th:replace标签替代。

进阶——fragment代码传参

使用fragment我们是可以在html代码中传参的,比如我们定义了一个top.html其中有一个“欢迎XXX”的提示,而这个人名XXX就是需要动态传递的,这样我们可以最大程度的完成代码的复用,这个时候就是一个很好的使用场景,我们需要这样做。(红色表示关键代码)

页面index.html代码:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:each="name,item:${names}">
    <span th:text="${item.count}"></span>
    <span th:text="${name}"></span>
</div>

<div th:replace="footer :: copyright"></div>

<div th:insert="footer :: about"></div>

<div th:include="footer :: links"></div>

<div th:replace="footer :: welcome('冯雪超')"></div>
</body>
</html>

 

 

页面footer.html

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div th:fragment="copyright">
    © 著作权归 冯雪超 所有
</div>

<div th:fragment="about">
    关于
</div>

<div th:fragment="links">
    CCTV
</div>

<div th:fragment="welcome(about)">
    <span th:text="|欢迎, ${about}|"></span>
</div>
</body>
</html>

 

 

最终的效果:

1.7 th:with 定义局部变量

页面代码:(红色表示关键代码)

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:each="name,item:${names}">
    <span th:text="${item.count}"></span>
    <span th:text="${name}"></span>
</div>

<div th:replace="footer :: copyright"></div>

<div th:insert="footer :: about"></div>

<div th:include="footer :: links"></div>

<div th:replace="footer :: welcome('冯雪超')"></div>

<div>
    <p th:with="temp='这是局部变量'">局部变量:[[${temp}]]</p>
</div>
</body>
</html>

 

页面输出结果:

1.8 th:remove 删除标签

th:remove用于html代码的删除,th:remove值有五个:

  • all 删除本段所有代码
  • body 删除主标签内的所有元素
  • tag 删除主标签,保留主标签所有的元素
  • all-but-first 保留主标签和第一个元素,其他全部删除
  • none 不删除任何标签

示例index.html代码如下:(红色表示关键代码)

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:each="name,item:${names}">
    <span th:text="${item.count}"></span>
    <span th:text="${name}"></span>
</div>

<div th:replace="footer :: copyright"></div>

<div th:insert="footer :: about"></div>

<div th:include="footer :: links"></div>

<div th:replace="footer :: welcome('冯雪超')"></div>

<div>
    <p th:with="temp='这是局部变量'">局部变量:[[${temp}]]</p>
</div>

<div>
    <div id="all" th:remove="all">
        <span>all</span>
        <span>1</span>
    </div>

    <div id="body" th:remove="body">
        <span>body</span>
        <span>2</span>
    </div>

    <div id="tag" th:remove="tag">
        <span>tag</span>
        <span>3</span>
    </div>


    <div id="all-but-first" th:remove="all-but-first">
        <span>all-but-first</span>
        <span>4</span>
    </div>

    <div id="none" th:remove="none">
        <span>none</span>
        <span>5</span>
    </div>
</div>
</body>
</html>

 

最终展示效果如下:

1.9 其他标签

  • th:style 定义样式 <div th:style="'color:'+${skinColor}">
  • th:onclick 点击事件 <input type="button" value=" Click " th:onclick="'onsub()'">
  • th:href 赋值属性href <a th:href="${myhref}"></a>
  • th:value 赋值属性value <input th:value="${user.name}" />
  • th:src 赋值src <img th:src="${img}" />
  • th:action 赋值属性action <form th:action="@{/suburl}">
  • th:id 赋值属性id <form id="${fromid}">
  • th:attr 定义多个属性 <img th:attr="src=@{/img/stone.jpg},alt=${alt}" />
  • th:object 定义一个对象 <div th:object="${user}">
  • ...

2.表达式使用

2.1 表达式概要

2.1.1 简单表达式

变量表达式:${...} 选择变量表达式:*{...} 消息表达式:#{...} 链接表达式:@{...} 片段表达:~{...}

2.1.2 数据的类型

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

2.1.3 文本操作

字符串拼接:+ 字面替换:|The name is ${name}|

2.1.4 算术运算

二进制运算符:+, -, *, /, % 减号(一元运算符):-

2.1.5 布尔运算

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

2.1.6 条件运算符

比较值:>, <, >=, <= 相等判断: ==, !=

2.1.7 条件判断

如果-然后:(if) ? (then) 如果-然后-否则:

(if) ? (then) : (else) 违约:(value) ?: (defaultvalue)

所有以上这些表达式都可以组合和嵌套,例如:

'User is of type ' + ({user.type} ?: 'Unknown'))

2.2 表达式使用实例

2.2.1 变量表达式 ${...}

变量表达式的使用,我们前面的代码已经见到了,$是我们平常开发中最常用的表达式,用于把后台Java类的动态数据,映射到页面。略

2.2.2 选择表达式 *{...}

选择表达式相当于选择了一个对象,在使用的时候不在需要这个对象的前缀,直接使用属性的key进行内容展示,代码如下:

Java代码:

package com.littlefxc.blog.controller;

import lombok.Data;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * @author fengxuechao
 */
@Controller
public class IndexController {


    @RequestMapping("/")
    public ModelAndView index(ModelAndView modelAndView) {
        modelAndView.setViewName("index");
        modelAndView.addObject("names", Arrays.asList("小王", "小明", "小红"));
        Goods goods = new Goods();
        goods.setName("iMac");
        goods.setPrice("20000");
        goods.setCreateTime(new Date());
        modelAndView.addObject("goods", goods);
        return modelAndView;
    }

    @Data
    public static class Goods{
        private String name;

        private String price;

        private Date createTime;
    }
}

 

 

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>

<div th:object="${goods}">
    <span th:text="${goods.name}"></span>
    <span th:text="*{price}"></span>
    <span th:text="${#dates.format(goods.createTime, 'yyyy-MM-dd HH:mm:ss')}"></span></div>
</body>
</html>

 

 

最终效果:

总结: *{price} = ${goods.price}只是省去了“goods.”前缀,效果都是一样的。

2.2.3 链接表达式 @{...}

用于转换url,代码如下:

<a th:href="@{/footer(id=666,name=fxc)}">链接</a>复制代码

最终呈现的效果:

<a href="/footer?id=666&name=fxc">链接</a>

链接表达式,可以传递参数,用逗号分隔。

服务器根相对路径:@{~/path/to/something}

2.2.4 文本操作

文本操作分为两个:文本拼加、文本替换

文本拼加:

<span th:text="'我叫'+${name}"></span>复制代码

文本替换:

文本替换的语法:|内容${tag}|

<span th:text="|我叫${name},是一名开发工程师。|"></span>复制代码

2.2.5 三元表达式

2.2.6 双括号作用

<p th:text="${val}">...</p><p th:text="${{val}}">...</p>复制代码

结果:

<p>1234567890</p><p>1,234,567,890</p>复制代码

2.2.7 嵌入文本标签

虽然标准的标签几乎可以满足所有的业务场景,但某些情况我们更喜欢直接写入HTML文本,例如:

<p>Hello, [[${name}]]</p>复制代码

嵌入文本有两种写法“[[...]]”和“[(...)]”,分别的作用就像th:text 和 th:utext 一样,例如:

<p>

    [[${name}]]

</p>

<p>

    [(${name})]

</p>复制代码

看到的效果是这样的:

2.3 表达式对象概述

表达式里面的对象可以帮助我们处理要展示的内容,比如表达式的工具类dates可以格式化时间,这些内置类的熟练使用,可以让我们使用Thymeleaf的效率提高很多。

2.3.1 表达式基本对象

  • #ctx: 操作当前上下文.
  • #vars: 操作上下文变量.
  • #request: (仅适用于Web项目) HttpServletRequest对象.
  • #response: (仅适用于Web项目) HttpServletResponse 对象.
  • #session: (仅适用于Web项目) HttpSession 对象.
  • #servletContext: (仅适用于Web项目) ServletContext 对象.

2.3.2 表达式实用工具类

  • #execInfo: 操作模板的工具类,包含了一些模板信息,比如:${#execInfo.templateName} .
  • #uris: url处理的工具
  • #conversions: methods for executing the configured conversion service (if any).
  • #dates: 方法来源于 java.util.Date 对象,用于处理时间,比如:格式化.
  • #calendars: 类似于 #dates, 但是来自于 java.util.Calendar 对象.
  • #numbers: 用于格式化数字.
  • #strings: methods for String objects: contains, startsWith, prepending/appending, etc.
  • #objects: 普通的object对象方法.
  • #bools: 判断bool类型的工具.
  • #arrays: 数组操作工具.
  • #lists: 列表操作数据.
  • #sets: Set操作工具.
  • #maps: Map操作工具.
  • #aggregates: 操作数组或集合的工具.

每个类中的具体方法,点击查看:www.thymeleaf.org/doc/tutoria…

三、Spring Boot 集成 Thymeleaf

3.1 Spring Boot 集成 Thymeleaf 分为四步:

  1. pom.xml 添加 Thymeleaf 模板引擎
  2. application.properties 配置 Thymeleaf 信息
  3. 创建controller类,编写代码
  4. 创建模板,编写html代码

接下来看具体的步骤。

3.1.1 pom.xml 添加 Thymeleaf 模板引擎

<!--thymeleaf模板-->

<dependency>

    <groupId>org.springframework.boot</groupId>

    <artifactId>spring-boot-starter-thymeleaf</artifactId>

</dependency>

3.1.2 application.properties 配置 Thymeleaf 信息

实际上在开发时,只需配置spring.thymeleaf.cache=false,其它都是默认配置,同时也不推荐自定义。

# 启用缓存:建议生产开启,

spring.thymeleaf.cache=false# 建议模版是否存在

spring.thymeleaf.check-template-location=true# Content-Type 值

spring.thymeleaf.servlet.content-type=text/html# 是否启用

spring.thymeleaf.enabled=true# 模版编码

spring.thymeleaf.encoding=utf-8# 应该从解析中排除的视图名称列表(用逗号分隔)

spring.thymeleaf.excluded-view-names=# 模版模式

spring.thymeleaf.mode=HTML5# 模版存放路径

spring.thymeleaf.prefix=classpath:/templates/# 模版后缀

spring.thymeleaf.suffix=.html复制代码

3.1.3 创建controller类,编写代码

我们在controller文件夹创建index.java,代码如下:

package com.littlefxc.blog.controller;

import lombok.Data;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.servlet.ModelAndView;

import java.util.Arrays;
import java.util.Date;
import java.util.HashMap;
import java.util.Map;

/**
 * @author fengxuechao
 */
@Controller
public class IndexController {


    @RequestMapping("/")
    public ModelAndView index(ModelAndView modelAndView) {
        modelAndView.setViewName("index");
        modelAndView.addObject("names", Arrays.asList("小王", "小明", "小红"));
        Goods goods = new Goods();
        goods.setName("iMac");
        goods.setPrice("20000");
        goods.setCreateTime(new Date());
        modelAndView.addObject("goods", goods);
        return modelAndView;
    }

    @Data
    public static class Goods{
        private String name;

        private String price;

        private Date createTime;
    }
}

 

 

关键代码解读:

  1. 不要用@ResponseBody注解:如果使用该注解,返回结果会直接输出,而不是使用模板引擎渲染
  2. 使用ModelAndView对象,指定视图名&添加视图对象

3.1.4 创建模板,编写html代码

我们在resources/templates下创建index.html,代码如下:

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>冯雪超的博客</title>
</head>
<body>
<div th:each="name,item:${names}">
    <span th:text="${item.count}"></span>
    <span th:text="${name}"></span>
</div>

<div th:replace="footer :: copyright"></div>

<div th:insert="footer :: about"></div>

<div th:include="footer :: links"></div>

<div th:replace="footer :: welcome('冯雪超')"></div>

<div>
    <p th:with="temp='这是局部变量'">局部变量:[[${temp}]]</p>
</div>

<div>
    <div id="all" th:remove="all">
        <span>all</span>
        <span>1</span>
    </div>

    <div id="body" th:remove="body">
        <span>body</span>
        <span>2</span>
    </div>

    <div id="tag" th:remove="tag">
        <span>tag</span>
        <span>3</span>
    </div>


    <div id="all-but-first" th:remove="all-but-first">
        <span>all-but-first</span>
        <span>4</span>
    </div>

    <div id="none" th:remove="none">
        <span>none</span>
        <span>5</span>
    </div>
</div>

<p></p>

<div th:object="${goods}">
    <span th:text="${goods.name}"></span>
    <span th:text="*{price}"></span>
    <span th:text="${#dates.format(goods.createTime, 'yyyy-MM-dd HH:mm:ss')}"></span></div>
</body>
</html>

 

footer.html

<!DOCTYPE html>
<html xmlns:th="http://www.thymeleaf.org">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
<div th:fragment="copyright">
    © 著作权归 冯雪超 所有
</div>

<div th:fragment="about">
    关于
</div>

<div th:fragment="links">
    CCTV
</div>

<div th:fragment="welcome(about)">
    <span th:text="|欢迎, ${about}|"></span>
</div>
</body>
</html>

 

 

效果如下:

四、参考资料

thymeleaf官方文档 Thymeleaf :www.thymeleaf.org/doc/tutoria…

thymeleaf官方文档 Spring + Thymeleaf :www.thymeleaf.org/doc/tutoria…

Thymeleaf 3.0 中文翻译https://github.com/jack80342/Thymeleaf-Reference-Guide

 

猜你喜欢

转载自blog.csdn.net/Little_fxc/article/details/83302884