SpringBoot同时支持多个视图解析器jsp+html+其他模版引擎!

你知道的越多,不知道的就越多,业余的像一棵小草!

你来,我们一起精进!你不来,我和你的竞争对手一起精进!

编辑:业余草

推荐:https://www.xttblog.com/?p=5140

SpringBoot同时支持多个视图解析器jsp+html+其他模版引擎!

有一个不算老的项目,经历过几波人迭代,源码维护的一塌糊涂。视图这一块,用的有 jsp,html,freemarker 等。视图不统一,导致启用 html 后,就不能访问 jsp 和 freemarker。这些具备互斥的因素,导致项目跑了 3 个,通过 Nginx 来适配。

今天,我给大家分享一种办法,让 SpringBoot 项目,同时支持多种模版引擎。

先说一下,这里没有新技术,新发明。只是根据 WebMvc 视图解析器的原理进行略微的改造。

我们都知道,在 SpringMVC 中有 3 个主要的类:DispatherServlet(前端控制器)、ViewResolver(视图解析器)、View(视图类)。

下面是一个简单的视图解析的流程。

由图可知,如果我们要支持多个视图,就需要配置多个视图解析器。比如:ThymeleafViewResolver、InternalResourceViewResolver(系统默认实现)、ContentNegotiatingViewResolver、BeanNameViewResolver等。

DispatcherServlet 在选择视图解析器的时候,以优先级为处理原则,此优先级是根据该 ViewResolver 实现 Ordered 接口或者使用 @Order 注解赋值,数字最小优先级越高为原则。

然后对应视图解析器会返回一个具体的 View 类。最终通过3、4步骤渲染成 HTML 或者是 XML 等视图内容。

下图就是具体的排序方法,viewResolvers 是一个 List 集合。

排完序后,匹配 View 对象的方法,遍历 viewResolvers,匹配到第一个 View 对象,则返回。

因此,当配置多个视图,需要同时支持多个视图时,就会发生一些 404(当我们配置多个视图解析器时,出现只支持一种视图解析器器,其他类型产生 404)。

以 ThymeleafViewResolver 为例,在 ThymeleafAutoConfiguration 中,可以看到它的 Order 为 Ordered.LOWEST_PRECEDENCE - 5。

而 SpringBoot 自动注入的 InternalResourceViewResolver 的优先级为最低级。

注意:这里指的是自动注入,就是在我们不经过任何改造的情况下,是属于最低级的。

@Bean
public InternalResourceViewResolver viewResolver() {
InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
    viewResolver.setPrefix("/WEB-INF/");
    viewResolver.setSuffix(".jsp");
    viewResolver.setOrder(0);
    viewResolver.setContentType("text/html;charset=UTF-8");
    return viewResolver;
}

由于,匹配到一个 View 对象,就立即返回。导致你不管配置了多少个视图解析器,返回的 View 可能都不是正确的,这就导致了 404 的出现。

那么有没有可能动态的调整排序,或者动态的指定视图解析器就成了关键。好在,SpringMVC 给我们留下了口子。我们只需要间的重写一个 InternalResourceView 即可。

public class HandleResourceViewExists extends InternalResourceView {
    @Override
    public boolean checkResource(Locale locale) {
        File file = new File(this.getServletContext().getRealPath("/") + getUrl());
        //判断页面是否存在
        return file.exists(); 
    }
}

通过上面的代码,我们先检查一下对应的模版文件是否存在。然后通过下面的代码,动态的调整视图解析器。

@Bean
public InternalResourceViewResolver htmlViewResolver() {
    InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
    viewResolver.setPrefix("/");
    //设置检查器
    viewResolver.setViewClass(HandleResourceViewExists.class); 
    viewResolver.setSuffix(".html");
    viewResolver.setOrder(0);
    viewResolver.setContentType("text/html;charset=UTF-8");
    return viewResolver;
}

@Bean
public InternalResourceViewResolver viewResolver() {
    InternalResourceViewResolver viewResolver = new InternalResourceViewResolver();
    //设置检查器
    viewResolver.setViewClass(HandleResourceViewExists.class);
    viewResolver.setPrefix("/WEB-INF/");
    viewResolver.setSuffix(".jsp");
    viewResolver.setOrder(0);
    viewResolver.setContentType("text/html;charset=UTF-8");
    return viewResolver;
}

更多视图的做法类似,我就不在贴代码了。这里给大家提供了一个思路,如果通过调试代码,发现问题,解决问题。

我是业余草,欢迎大家关注,留言评论!

业余草 CSDN认证博客专家 Java Go/GoLang Redis
如今的编程是一场程序员和上帝的竞赛,程序员要开发出更大更好、傻瓜都会用到软件。而上帝在努力创造出更大更傻的傻瓜。目前为止,上帝是赢的。个人网站:www.xttblog.com。个人微信:codedq或dart996
已标记关键词 清除标记
©️2020 CSDN 皮肤主题: 大白 设计师:CSDN官方博客 返回首页