SpringBoot 静态资源导入及首页设置问题
作者:風栖祈鸢 发布时间:2023-11-26 22:45:07
本节了解一下 SpringBoot 中 Web 开发的静态资源导入和首页设置,对应 SpringBoot-03-Web 项目。
1. 静态资源导入
在 Web 开发过程中,我们需要接触许多的静态资源,如 CSS、JS、图片等;在之前的开发过程中,这些资源都放在 Web 的目录下,用到的时候按照对应路径访问即可。不过在 SpringBoot 项目中,没有了 Web 的目录,那这些静态资源该放到哪里去,又要如何访问呢?
由于是 Web 应用中的配置,所以查看对应的自动配置类 WebMvcAutoConfiguration
,可以看到处理资源的方法 addResourceHandlers
@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
if (!this.resourceProperties.isAddMappings()) {
logger.debug("Default resource handling disabled");
return;
}
addResourceHandler(registry, "/webjars/**", "classpath:/META-INF/resources/webjars/");
addResourceHandler(registry, this.mvcProperties.getStaticPathPattern(), (registration) -> {
registration.addResourceLocations(this.resourceProperties.getStaticLocations());
if (this.servletContext != null) {
ServletContextResource resource = new ServletContextResource(this.servletContext, SERVLET_LOCATION);
registration.addResourceLocations(resource);
}
});
}
其中,this.resourceProperties.isAddMappings()
的作用为判断是否在配置文件中指定了资源的访问路径,若指定了则此方法不用生效,直接返回;若未指定则继续执行方法,去默认的位置查找资源。
1.1 WebJars
WebJars 是前端资源的 Jar 包形式,让我们可以通过 Jar 包的形式使用前端的框架、组件。
WebJars 网站:https://www.webjars.org/。
为什么要使用 WebJars?
我们在开发 Java web 项目的时候会使用像 Maven,Gradle 等构建工具以实现对 jar 包版本依赖管理,以及项目的自动化管理,但是对于 JS,Css 等前端资源包,我们只能采用拷贝到 webapp 目录下的手工方式,这样做就无法对这些资源进行依赖管理,而且容易导致文件混乱、版本不一致等问题。WebJars 就提供给我们这些前端资源的 jar 包形式,我们就可以进行依赖管理。
如要使用到 JQuery 时,按照之前的做法,我们要去网上下载 JQuery 的 JS 文件,把它放到 web 目录下的 statics/js 下(之前用 AJAX 的时候就是这么干的);但现在,我们可以采用 WebJars 的方式。
首先在 WebJars 网站中找到 JQuery 的 Maven 坐标,把它放到项目的 pom 文件中
<dependency>
<groupId>org.webjars</groupId>
<artifactId>jquery</artifactId>
<version>3.6.0</version>
</dependency>
引入后,在项目的 External Libaries 中就可以看到 org.webjars:jquery:3.6.0 了!
那么我们要怎么访问到它呢?在上面的源码中其实就已经给出了路径
addResourceHandler(registry, "/webjars/**", "classpath:/META-INF/resources/webjars/");
这行代码将 /webjars/
下的所有访问都映射为了 classpath:/META-INF/resources/webjars/
,即我们只需要通过 /webjars/
就可以找到类路径下的 /jquery/3.6.0/jquery.js
文件了!
运行项目,在浏览器中输入 http://localhost:8080/webjars/jquery/3.6.0/jquery.js
,确实显示出了 jquery.js 文件!
以 WebJars 方式引入的文件,都符合上图中的结构,即能通过 classpath:/META-INF/resources/webjars/
路径访问到,这样代码中的设置和外部文件就联系起来了!
1.2 staticPathPattern
回到源码中,这个方法的三句话还有最后一句(虽然很长但确实是一句)
addResourceHandler(registry, this.mvcProperties.getStaticPathPattern(), (registration) -> {
registration.addResourceLocations(this.resourceProperties.getStaticLocations());
if (this.servletContext != null) {
ServletContextResource resource = new ServletContextResource(this.servletContext, SERVLET_LOCATION);
registration.addResourceLocations(resource);
}
});
这就有点复杂了(之前版本的源码倒还好理解一点),不过可以看到获取静态路径 getStaticPathPattern()
方法,点进去
public String getStaticPathPattern() {
return this.staticPathPattern;
}
这个方法直接返回了 staticPathPattern
,继续点
/**
* Path pattern used for static resources.
*/
private String staticPathPattern = "/**";
到这就明白了,其实就是默认的静态资源路径!这个路径也可以通过 spring.mvc 去设置,在未设置的情况在,它就是项目下的所有路径 /**
!
然后在 Web 属性类 WebProperties 中有一个资源类 Resource
,它也设置了4个路径(跳跃的有点大,先看着吧),其中
public static class Resources {
private static final String[] CLASSPATH_RESOURCE_LOCATIONS = { "classpath:/META-INF/resources/",
"classpath:/resources/", "classpath:/static/", "classpath:/public/" };
...
}
classpath:/META-INF/resources/
即上面的 WebJars 路径classpath:/resources/
即resources/resources/
路径classpath:/static/
为resources/static/
路径classpath:/public/
为resources/public
路径
即所有通过 /**
(未配置情况下)的访问请求,都会在这四个路径中寻找静态资源!
默认的 resource 中只有 static 一个目录,这里把上面的目录都创建一下,且放入一个测试用的 js 文件
此时运行项目,访问 http://localhost:8080/public.js
、http://localhost:8080/resources.js
、http://localhost:8080/static.js
,都可以显示出对应的 js 文件内容!
注意:如果三个目录下的文件有重名的情况,则优先级为 CLASSPATH_RESOURCE_LOCATIONS
数组的顺序,可以理解为如果在前面的路径中找到了就不找后面的了!
2. 首页设置
和上面一样,先找到对应的源码
@Bean
public WelcomePageHandlerMapping welcomePageHandlerMapping(ApplicationContext applicationContext,
FormattingConversionService mvcConversionService, ResourceUrlProvider mvcResourceUrlProvider) {
WelcomePageHandlerMapping welcomePageHandlerMapping = new WelcomePageHandlerMapping(
new TemplateAvailabilityProviders(applicationContext), applicationContext, getWelcomePage(),
this.mvcProperties.getStaticPathPattern());
welcomePageHandlerMapping.setInterceptors(getInterceptors(mvcConversionService, mvcResourceUrlProvider));
welcomePageHandlerMapping.setCorsConfigurations(getCorsConfigurations());
return welcomePageHandlerMapping;
}
很长也很复杂,不过只需要关注里面的 getWelcomePage()
方法,点进去看看
private Resource getWelcomePage() {
for (String location : this.resourceProperties.getStaticLocations()) {
Resource indexHtml = getIndexHtml(location);
if (indexHtml != null) {
return indexHtml;
}
}
ServletContext servletContext = getServletContext();
if (servletContext != null) {
return getIndexHtml(new ServletContextResource(servletContext, SERVLET_LOCATION));
}
return null;
}
private Resource getIndexHtml(String location) {
return getIndexHtml(this.resourceLoader.getResource(location));
}
private Resource getIndexHtml(Resource location) {
try {
Resource resource = location.createRelative("index.html");
if (resource.exists() && (resource.getURL() != null)) {
return resource;
}
}
catch (Exception ex) {
}
return null;
}
这三个方法是逐层调用的关系(虽然我也不知道为什么要这么干),不过可以知道,其中的 location
就是上面的三个目录 resources
、static
、public
,默认的首页是 index.html
。也就是说,如果这三个目录下存在 index.html
文件,那么它就是默认的首页!演示就省略了,反正也不是什么难点!
3. 总结
本节主要是从源码的角度,研究了一下静态资源导入和首页设置的问题。其实学习结论很简单,但从源码出发思考问题的思想,是不容易学习的。
来源:https://blog.csdn.net/qq_43560701/article/details/120452787


猜你喜欢
- SpringBoot集成Redis 1.添加redis依赖<dependency> <groupId
- Quote在学习 Kotlin 的过程中,对 Kotlin 的类型系统产生了好奇,Kotlin 是否存在类似于 Java 中 Object
- 一:深入解析JSTL标签库 1、什么是JSTL标签库①Java Standard Tag Lib(Java标准的标签库)。②JST
- 项目结构:效果图一:效果图二:效果图三:关于闹钟:Alarm hongten v1.0 使用说明: &nbs
- 本文实例讲述了C#使用Object类实现栈的方法。分享给大家供大家参考,具体如下:Stack类的代码:using System;using
- 好几天没有跟进Spring MVC的学习了,之前看了点源码都忘的差不多了。这次就跟着之前的问题,继续总结下Spring MVC中的小知识。u
- 功能函数// 图像旋转void Rotate(const cv::Mat &srcImage, cv::Mat &dstIm
- 文件上传是开发中十分常见的功能,在servlet3.0之前,实现文件上传需要使用一些插件技术,比如:commons-fileuploadsm
- MultipartResolver和ServletFileUpload冲突如果同时使用了MultipartResolver 和Servlet
- 普通的公有继承class test1{public: test1(int i) :num1(i) {}privat
- 函数指针是指向函数的指针,指针函数是指一个函数的返回值是一个指针,但下面的几道题还是感觉很迷惑。各位能否讲的详细点呢?(1) float(*
- 前言今天是2021LOL全球总决赛,一直不被大家看好的EDG冲到了决赛对战韩国队的DK,可以说EDG面对如此强大的对手,想赢是比较难的,为了
- 集合的创建和遍历方式集合只要包含List和Set在宽泛一些就包括Map这种键值对类型的数据结构List,Set和Map在java当中都是接口
- 项目涉及蓝牙通信,所以就简单的学了学,下面是自己参考了一些资料后的总结,希望对大家有帮助。 以下是开发中的几个关键步骤:1、首先开
- 游戏音效就是我们在玩游戏时出现的音乐,这个也是每个游戏必备的一部分,但有是你做游戏的背景音乐有间断的感觉的话,我们可以用getCurrent
- Kotlin中StateFlow的使用StateFlow 是 Flow 的实现,是一个特殊的流,默认的 Flow 是冷流,而StateFlo
- 本文以一个asp.net程序为例讲述了Repeater中添加按钮实现点击按钮获取某一行数据的方法,分享给大家供大家参考借鉴之用。具体步骤如下
- 本文汇总了Spring的常用注解,以方便大家查询和使用,具体如下:使用注解之前要开启自动扫描功能其中base-package为需要扫描的包(
- 为了方便客户抓取Log,现通过TCP协议连接指定服务器,传输指定内容,定义指定目录,IP,PORT字段接收参数。直接上代码 public s
- 本文实例讲述了android编程之ip2id程序。分享给大家供大家参考。具体分析如下:一、说明:公司一个项目中需要给一系列网络设备分配id号