最近团队的一个项目在重构,希望引入Thymeleaf减少页面端的代码复杂性。在重构过程中,发现html文件需要保存在多个不同的目录中,但Thymeleaf缺省的实现不支持这种方式。

1        背景

Maven项目,前端使用SpringMVC,没有使用任何模板引擎。所有的页面内容,都是通过静态HTML+AJAX+JSON形式实现。

1.1     项目结构

Html文件,通过mvc:resource 定义路径。

1.1.1            Html保存路径

/hardess_finance/src/main/webapp/WEB-INF/htmls

在该目录放一个Demo文件/demo/hello.html

1.1.2            Spring配置文件

<mvc:resources mapping="/**/**.html" location="/WEB-INF/htmls/"/>

项目启动后,浏览器访问 http://localhost:8080/demo.hello.html,就可以访问到demo文件。

1.2     添加Thymeleaf支持

Spring Boot 项目缺省使用Thymeleaf模板,但普通SpringMVC项目,需要手工添加支持。大致步骤包括:

1.2.1            Pom.xml增加thymeleaf dependency

<dependency>

<groupId>org.thymeleaf</groupId>

<artifactId>thymeleaf</artifactId>

<version>3.0.6.RELEASE</version>

</dependency>

<dependency>

<groupId>org.thymeleaf</groupId>

<artifactId>thymeleaf-spring3</artifactId>

<version>3.0.6.RELEASE</version>

</dependency>

1.2.2            修改Spring Config文件

<bean id="templateResolver"

class="org.thymeleaf.spring3.templateresolver.SpringResourceTemplateResolver">

<property name="prefix"><value>/WEB-INF/html/</value></property>

<property name="suffix"><value>.html</value></property>

<property name="templateMode"><value>HTML</value></property>

<property name="characterEncoding"><value>UTF-8</value></property>

<property name="cacheable" value="false"/>

</bean>

<bean id="templateEngine" class="org.thymeleaf.spring3.SpringTemplateEngine">

<property name="templateResolver" ref="templateResolver" />

</bean>

<bean class="org.thymeleaf.spring3.view.ThymeleafViewResolver">

<property name="templateEngine" ref="templateEngine" />

<property name="characterEncoding" value="UTF-8"/>

</bean>

<property name="prefix"><value>/WEB-INF/html/</value></property>这个参数,指定的就是html文件的保存路径。

1.2.3            添加Thymeleaf Controller

在代码中增加一个Controller

@Controller

public class ThymeleafCommonController {

@RequestMapping(value = { "/**/**.html" })

public ModelAndView index(HttpServletRequest request) {

return new ModelAndView();

}

}

至此,项目重新启动后,所有html的访问,都已经通过Thymeleaf引擎,html中能够使用 th:text 等各种thymeleaf语法。

1.3     项目重构希望添加另一个html保存路径

在重构过程中,希望将html文件保存到新的目录 src/main/resources/templates目录,原因有二:

1、              这是Spring Boot项目的缺省模板路径,适应将来可能升级到SpringBoot的需求。

2、              重构后的代码,希望同原目录有所区分,简化开发复杂度。

1.3.1            简单尝试

在 mvc:resources标签中,location可以是用逗号隔开的多个路径,如

<mvc:resources mapping="/scripts/**"

location="/WEB-INF/scripts/, classpath:/static/scripts/"/>

因此,尝试在spring config配置文件中,尝试修改配置

<property name="prefix">

<value>/WEB-INF/html/,classpath:/templates/</value>

</property>

重启后测试,发现项目无法工作,原有的界面都无法加载了。

1.3.2            原因

Debug了一下thymeleaf的相关源码,发现它使用下面的语句生成最终的完整路径名,并没有判断 prefix 是否是逗号分隔的数组。

AbstractConfigurableTemplateResolver.computeResourceName(…)

return prefix + unaliasedName + suffix;

2        Thymeleaf源码解读

解读Thymeleaf的源代码,发现几个相关类

2.1     相关类结构

2.2     final computeTemplateResource()

这个函数会读取配置的prefix,并调用后续方法生成 resource name。注意,这个方式是 final ,无法重载。

@Override

protected final ITemplateResource computeTemplateResource(

final IEngineConfiguration configuration, final String ownerTemplate,

final String template, final Map<String, Object> templateResolutionAttributes) {

final String resourceName =

computeResourceName(configuration, ownerTemplate, template, this.prefix,

this.suffix, this.forceSuffix, this.templateAliases, templateResolutionAttributes);

return computeTemplateResource(configuration, ownerTemplate, template, resourceName,

this.characterEncoding, templateResolutionAttributes);

}

标红部分读取prefix参数值。

2.3     computeResourceName()

实际生成resource name(代码有删减,只保留核心部分)

protected String computeResourceName(

final IEngineConfiguration configuration, final String ownerTemplate, final String template,

final String prefix, final String suffix, final boolean forceSuffix,

final Map<String, String> templateAliases, final Map<String, Object> attributes) {

String unaliasedName = templateAliases.get(template);

if (unaliasedName == null) {

unaliasedName = template;

}

// hasPrefix && shouldApplySuffix

return prefix + unaliasedName + suffix;

}

2.4     computeResolvable()

判断资源文件是否可用。

if (this.resolvablePatternSpec.isEmpty()) {

return true;

}

return this.resolvablePatternSpec.matches(template);

这个代码,实际没有校验html文件是否存在,只要语法不出错即可。当系统定义了多个ITemplateResolver时,引擎回依次调用每个实例的computeResolvable()方法,如果返回null,则依次检查下一个resolver,直到得到一个非空值。

3        解决方案

基于前面的代码分析,要解决我们的需求,首先我们需要解决的是判断资源文件是否真实存在。

3.1     判断文件是否存在

通过Spring项目的ApplicationContext判断文件是否存在的代码片段。

resolvable = false;

Resource resource = applicationContext.getResource(location);

if (resource != null && resource.exists()) {

resolvable = true;

}

为了验证解决方案的可行性,增加了一个新的html文件在 src/main/resources/templates/demo/world.html

3.2     方案一:定义多个 TemplateResolver

3.2.1            Custom TemplateResolver

Spring提供的实现类SpringResourceTemplateResolver,代码比较简单,我选择直接替换该类,而不是从它继承而来。

public class CodeStoryTemplateResolver extends AbstractConfigurableTemplateResolver

implements ApplicationContextAware {

private ApplicationContext applicationContext = null;

public CodeStoryTemplateResolver() {

super();

}

public void setApplicationContext(final ApplicationContext applicationContext)

throws BeansException {

this.applicationContext = applicationContext;

}

@Override

protected boolean computeResolvable(

IEngineConfiguration configuration, String ownerTemplate,

String template, Map<String, Object> templateResolutionAttributes) {

boolean resolvable = super.computeResolvable(configuration, ownerTemplate,

template, templateResolutionAttributes);

    if (resolvable) {

      // 判断文件是否存在

      resolvable = false;

      String pathName = getPrefix() + template + getSuffix();

      Resource resource = applicationContext.getResource(pathName);

      if (resource != null && resource.exists()) {

        resolvable = true;

      }

    }

return resolvable;

}

@Override

protected ITemplateResource computeTemplateResource(

final IEngineConfiguration configuration, final String ownerTemplate,

final String template, final String resourceName, final String characterEncoding,

final Map<String, Object> templateResolutionAttributes) {

return new SpringResourceTemplateResource(

this.applicationContext, resourceName, characterEncoding);

}

}

3.2.2            修改Spring配置

<bean id="webinfoTemplateResolver" class="....CodeStoryTemplateResolver">

<property name="prefix">

<value>/WEB-INF/</value>

</property>

<property name="suffix">

<value>.html</value>

</property>

<property name="templateMode">

<value>HTML</value>

</property>

<property name="characterEncoding">

<value>UTF-8</value>

</property>

<property name="cacheable" value="false"/>

</bean>

<bean id="webinfoTemplateResolver" class="....CodeStoryTemplateResolver">

<property name="prefix">

<value>classpath:/templates/</value>

</property>

<property name="suffix">

<value>.html</value>

</property>

<property name="templateMode">

<value>HTML</value>

</property>

<property name="characterEncoding">

<value>UTF-8</value>

</property>

<property name="cacheable" value="false"/>

</bean>

<bean id="templateEngine" class="org.thymeleaf.spring3.SpringTemplateEngine">

<property name="templateResolvers">

<set>

<ref bean="webinfoTemplateResolver" />

<ref bean="classpathTemplateResolver" />

</set>

</property>

</bean>

<bean class="org.thymeleaf.spring3.view.ThymeleafViewResolver">

<property name="templateEngine" ref="templateEngine" />

<property name="characterEncoding" value="UTF-8"/>

</bean>

重启系统测试,/demo/hello.html和/demo/world.html都能正常访问。

3.3     方案二:一个TemplateResolver支持prefixes

3.3.1            Custom TemplateResolver

理想的方案,是重载函数computeTemplateResource(),但这个函数被定义为final,无法重载,只好退而求其次选择重载computeResourceName()。在这个函数中判断是否定义了prefixes参数,如果是一次调用父类的computeResourceName()并判断资源是否存在。

public class CodeStoryTemplateResolver extends AbstractConfigurableTemplateResolver

implements ApplicationContextAware {

private ApplicationContext applicationContext = null;

public void setApplicationContext(final ApplicationContext applicationContext)

throws BeansException {

this.applicationContext = applicationContext;

}

  private String prefixes = null;

  public CodeStoryTemplateResolver() {

    super();

  }

  public final String getPrefixes() {

    return this.prefixes;

  }

  public final void setPrefixes(final String prefixes) {

    this.prefixes = prefixes;

  }

protected String computeResourceName(final IEngineConfiguration configuration,

final String ownerTemplate, final String template, final String prefix,

final String suffix, final boolean forceSuffix,

final Map<String, String> templateAliases,

final Map<String, Object> templateResolutionAttributes) {

String resourceName = null;

    String[] prefixes = null;

    if (!StringUtils.isEmptyOrWhitespace(getPrefixes())) {

      prefixes = getPrefixes().split(",");

    } else if (!StringUtils.isEmptyOrWhitespace(getPrefix())) {

      prefixes = new String[] { getPrefix() };

    } else {

      prefixes = new String[] { "" };

    }

    for (String onePrefix : prefixes) {

      onePrefix = StringUtil.trimLeft(StringUtil.trimRight(onePrefix));

      resourceName = super.computeResourceName(configuration, ownerTemplate,

             template, onePrefix, suffix,

        forceSuffix, templateAliases, templateResolutionAttributes);

      Resource resource = applicationContext.getResource(resourceName);

      if (resource != null && resource.exists()) {

        break;

      } else {

        resourceName = null;

      }

    }

return resourceName;

}

@Override

protected ITemplateResource computeTemplateResource(

final IEngineConfiguration configuration, final String ownerTemplate,

final String template, final String resourceName,final String characterEncoding,

final Map<String, Object> templateResolutionAttributes) {

return new SpringResourceTemplateResource(this.applicationContext,

resourceName, characterEncoding);

}

}

3.3.2            修改Spring配置

<bean id="multiTemplateResolver" class="....CodeStoryTemplateResolver">

<property name="prefixes">

<value>/WEB-INF/,classpath:/templates/</value>

</property>

<property name="suffix">

<value>.html</value>

</property>

<property name="templateMode">

<value>HTML</value>

</property>

<property name="characterEncoding">

<value>UTF-8</value>

</property>

<property name="cacheable" value="false"/>

</bean>

<bean id="templateEngine" class="org.thymeleaf.spring3.SpringTemplateEngine">

<property name="templateResolvers">

<set>

<ref bean="multiTemplateResolver" />

</set>

</property>

</bean>

<bean class="org.thymeleaf.spring3.view.ThymeleafViewResolver">

<property name="templateEngine" ref="templateEngine" />

<property name="characterEncoding" value="UTF-8"/>

</bean>

重启系统测试,/demo/hello.html和/demo/world.html都能正常访问。

3.4     啰嗦几句

两种方案区别不大,都是尝试增加文件判断,便于Thymeleaf找到真正的html所在路径。配置上,第二种方案相对简单一点。

性能方面,没有做仔细测试,估计比Spring缺省的TemplateResolver会慢一些。

当然,这个方案有点多次一举,最简单的处理方式,把目录src/main/webapps/WEB-INF/htmls 移动到 src/main/resources/templates即可。

Thymeleaf引擎支持Multi Prefix的更多相关文章

  1. spring mvc中添加对Thymeleaf的支持

    一.下载Thymeleaf 官方下载地址:https://dl.bintray.com/thymeleaf/downloads/thymeleaf/ 我下载的是最新的3.0.11版本 把包里的jar包 ...

  2. Atitti 存储引擎支持的国内点与特性attilax总结

    Atitti 存储引擎支持的国内点与特性attilax总结 存储引擎处理的事情: · 并发性:某些应用程序比其他应用程序具有很多的颗粒级锁定要求(如行级锁定). · 事务支持:并非所有的应用程序都需要 ...

  3. FBReader阅读引擎支持的功能

    "三十年河东,三十年河西"是一句民间谚语,它的来源是:从前黄河河道不固定,经常会改道(历史上无数次发生).某个地方原来在河的东面,若干年后,因黄河水流改道,这个地方会变为在河的西面 ...

  4. (转)支持Multi Range Read索引优化

    支持Multi Range Read索引优化 原文:http://book.51cto.com/art/201701/529465.htm http://book.51cto.com/art/2016 ...

  5. MyISAM、InnoDB、Memory这3个常用引擎支持的索引类型

    表格对比了MyISAM.InnoDB.Memory这3个常用引擎支持的索引类型: 索引 MyISAM引擎 InnoDB引擎 Memory引擎 B-Tree索引 支持 支持 支持 HASH索引 不支持 ...

  6. springboot框架中集成thymeleaf引擎,使用form表单提交数据,debug结果后台获取不到数据

    springboot框架中集成thymeleaf引擎,使用form表单提交数据,debug结果后台获取不到数据 表单html: <form class="form-horizontal ...

  7. Flutter 多引擎支持 PlatformView 以及线程合并解决方案

    作者:字节移动技术-李皓骅 摘要 本文介绍了 Flutter 多引擎下,使用 PlatformView 场景时不能绕开的一个线程合并问题,以及它最终的解决方案.最终 Pull Request 已经 m ...

  8. Spring Boot thymeleaf模版支持,css,js等静态文件添加

    Thymeleaf引入 Thymeleaf是一个Java模板引擎开发库,可以处理和生成HTML.XML.JavaScript.CSS和文本,在Web和非Web环境下都可以正常工作. 1.添加依赖包 & ...

  9. SpringBoot入门篇--Thymeleaf引擎模板的基本使用方法

    我们在使用SpringBoot框架的时候在前面已经介绍了Thymelea引擎模板,因为SpringBoot对JSP惨不忍睹的支持.那我们在使用引擎模板对前端页面进行渲染能够返回的情况下我们怎么才能在静 ...

随机推荐

  1. 在LwIP 协议栈移植 Snap 7

    本文欢迎引用,转载. 引用,转载请标明出处! 调试完毕源码将上传到GitHub 为了嵌入式系统与STEP 7 PLC 通过S7 协议通讯,尝试移植 Snap 7 到STM32F407 cpu 上. 今 ...

  2. 本地git部署web连接azure的git存储库

    ​​​本地git部署web 创建本地存储仓库 输入以下命令创建git本地仓库(会在当前目录下生产一个.git的目录) git init 然后提交内容 在git仓库所在的目录下存放好需要的网页文件 将文 ...

  3. 免费人脸识别APi

    今天对应一些免费的人脸识别的api 做了一下简单的对比,觉得百度开发出来的人脸识别接口还是最符合的我的要求,简单易用,容易上手. 据说百度的一些门禁也使用上了人脸识别的功能了,功能很强大,而且能识别出 ...

  4. 爬虫day 04(通过登录去爬虫 解决django的csrf_token)

    #通过登录去爬虫 #首先要有用户名和密码 import urllib.request import http.cookiejar from lxml import etree head = { 'Co ...

  5. Handlebars 和 angularjs 之间的区别

    handlebarsjs算不上框架,只是一种js模板引擎,是模板库,模板库的主要作用是:你想要生成某一大片有一定规律的界面,比如商品详情,不同商品之间差的只是名称,价格,图片,介绍这些,但是结构一样的 ...

  6. centos 下安装pptp (vpn) 的方法

    废话少说     01 02 03 04 05 06 07 08 09 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 3 ...

  7. 基于Jmeter的接口自动化测试实践

    在去年实施了一年的三端(PC.无线M站.无线APP[Android.IOS])后,今年7月份开始,我们开始进行接口自动化的实施,目前已完成了整个框架的搭建以及接口的持续测试集成.今天做个简单的分享. ...

  8. JavaScript实现AOP(面向切面编程)

    什么是AOP? AOP(面向切面编程)的主要作用是把一些跟核心业务逻辑模块无关的功能抽离出来,这些跟业务逻辑无关的功能通常包括日志统计.安全控制.异常处理等.把这些功能抽离出来之后, 再通过" ...

  9. C#真的过时了吗?

    现在有一种言论:C#过时了!!! 有人说现在是BS的时代,C#开发BS网站的那一套,相对于Java.PHP来说,效率太低了! 有人说现在是移动互联网时代,C#作为微软主推的语言,无法开发移动应用成为其 ...

  10. java实现导出Excel(跨行,跨列)

    先来个最终结果样式: 第一步: 传参,后期可根据自己需要进行调整.我这里需要的是 quarter 代表季度 dptid 部门编号根据接受过来的参数进行文档命名. UserInfo userInfo=( ...