spring boot中配置共用Thymeleaf和Jsp

一、Spring Boot中新增了包括Thymeleaf在内的诸多模版引擎的自动装载,但奇怪的是,自动注入的ThymeleafviewResolver也会匹配到寻找Jsp视图的请求,显然ThymeleafviewResolver找不到对应的Thymeleaf视图后接着返回了error错误页面,于是排在后面用来解析Jsp视图的InternalResourceViewResolver就失效了。

二、于是我通过@SpringBootApplication(exclude = {ThymeleafAutoConfiguration.class})不让Spring Boot自动装载Thymeleaf模板引擎,参照网上的方法手动去装载,但发现一个很奇怪的情况,把templateResolver的前缀设置为templateResolver.setPrefix("classpath:/templates/");无法找到该路径下的html资源,只有设置为templateResolver.setPrefix("/WEB-INF/views/");才能找到对应路径下的html资源,而“/WEB-INF/views/”其实是Jsp视图的InternalResourceViewResolver中设置的前缀,仿佛是ThymeleafviewResolver自己的templateResolver前缀无效,只有和InternalResourceViewResolver的viewResolver前缀一致时才能找到对应的html资源。

三、下一步研究ThymeleafviewResolver自己的templateResolver为什么会失效,通过ThymeleafAutoConfiguration类入手逐渐找到了Spring Boot设置Thymeleaf的templateResolver的对应类AbstractTemplateResolverConfiguration下的defaultTemplateResolver方法,反复对比了很久后才发现官方是用SpringResourceTemplateResolver来设置,而我在网上搜的是用ServletContextTemplateResolver来配置,再去网上搜索,发现在thymeleaf3.0版本中已经使用SpringResourceTemplateResolver代替了ServletContextTemplateResolver。


四、继续查询一些资料后才发现,ServletContextTemplateResolver失效的是因为需要ServletContext把它作为参数去传递给Spring,但目前的Spring+Thymeleaf机制中好像并没有实现这样的传递,于是我们需要使用提供了类似机制的SpringResourceTemplateResolver,从而能够基于Spring的资源解析功能寻找资源,下图截取于https://github.com/thymeleaf/thymeleaf/issues/419。


五、一直以为在Spring MVC中,若ThymeleafviewResolver找不到对应的Thymeleaf视图应该继续匹配后面的viewResolver,直到全部匹配完还找不到才返回error页面,但现实情况明显不是这样,断点调试后发现,只要匹配上了一个viewResolver,若按照此viewResolver找不到对应的资源,那么就会返回error页面,此时若要实现共用Thymeleaf和Jsp,可以想到三种思路。

1、viewResolver按照ViewNames来匹配,即配置Thymeleaf和Jsp的viewResolver只可匹配上各自的视图名称,目前采用这种方式来转发;

2、研究viewResolver的依次匹配机制,使得匹配上的viewResolver找不到对应的资源时继续匹配剩下的viewResolver,未完待续;

3、配置Multiple Dispatcher Servlets来拦截不同请求。

a) 参考http://bitswithbytes.com/2016/10/13/multiple-dispatcher-servlets-in-spring-boot-web-app/,需要做两点变动,一是将Spring Boot自动注册的DispatcherServlet做如下配置,二是手动注入新的DispatcherServlet。


但尝试之后发现Spring Boot自动注册的DispatcherServlet无法将请求转发到对应的@RequestMapping响应方法上,说明这个配置是有问题的,有待进一步研究。

b) 配置Spring Boot自动注册的DispatcherServlet只拦截“*.jsp”的请求,自己手动注入新的DispatcherServlet(名为thymeleafServlet)只拦截“*.thymeleaf”的请求,可能会存在两个难点,一是如何配置Spring Boot自动注册的DispatcherServlet只拦截“*.jsp”的请求,在spring boot默认的配置参数中并没有这个值,参见http://docs.spring.io/spring-boot/docs/current/reference/html/common-application-properties.html,这时可能就需要继承自动注册DispatcherServlet来设置;二是如何将自己手动注册的thymeleafServlet和Thymeleaf的viewResolver进行绑定,未完待续




评论 6
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值