• SpringBoot访问静态资源报404——记录一次调试过程与解决方案


    SpringBootWeb项目中,默认的静态资源路径有下列4种:

    1. classpath:/META-INF/resources
    2. classpath:/resources
    3. classpath:/static
    4. classpath:/public**
      項目路徑

    默认情况下,浏览器中输入localhost:port/index.html将会成功访问到该资源文件[静态资源路径符合第三点],但是却出乎意料,报了404找不到该资源。
    在这里插入图片描述
    在调试之前需要回顾一下SpringMVC中DispatchServelet的工作流程,如何处理一个请求并将结果成功返回给浏览器
    DispatchServelet工作流程

    不得不提的是,DispatchServelet其中一个最重要的一个步骤就是从HandlerMapping中获得处理请求的Handler
    在这里插入图片描述
    调试DispatchServelet时,可得知获得该Handler为空,所以才会进入noHandlerFound()方法,对请求进一步处理

    	protected void noHandlerFound(HttpServletRequest request, HttpServletResponse response) throws Exception {
    		if (pageNotFoundLogger.isWarnEnabled()) {
    			pageNotFoundLogger.warn("No mapping for " + request.getMethod() + " " + getRequestUri(request));
    		}
    		if (this.throwExceptionIfNoHandlerFound) {
    			throw new NoHandlerFoundException(request.getMethod(), getRequestUri(request),
    					new ServletServerHttpRequest(request).getHeaders());
    		}
    		else {
    		    //Http请求返回404
    			response.sendError(HttpServletResponse.SC_NOT_FOUND);
    		}
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13

    所以这里返回404的原因就是对输入的请求无法找到一个合适的处理器,获得handler的源码如下:

    	protected HandlerExecutionChain getHandler(HttpServletRequest request) throws Exception {
    		if (this.handlerMappings != null) {
    			for (HandlerMapping mapping : this.handlerMappings) {
    				HandlerExecutionChain handler = mapping.getHandler(request);
    				if (handler != null) {
    					return handler;
    				}
    			}
    		}
    		return null;
    	}
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11

    在这里插入图片描述
    无法从注册的HangdlerMappings列表中通过一定的映射关系获得HandlerExecutionChain【handler和拦截器的结合】

    SimpleUrlHandlerMapping : 它内部有一个Map urlMap,存放着各个url对应的handler,适用性最强的Handler Mapping类,允许明确指定URL模式和Handler的映射关系。处理静态资源的就是从这个SimpleUrlHandlerMapping 中获得Handler

    继续调试可知

    在这里插入图片描述

    SimpleUrlHandlerMapping 的UrlMap存放的映射无法支持找到static/index.html,允许访问的资源路径仅仅只有classpath:/META-INF/resources/ 和classpath:/META-INF/resources/webjars/
    根本原因找到了,我曾经重写了资源路径的方法,所以才无法访问到static文件夹下的index.html

    public class SwaggerConfig implements WebMvcConfigurer {
        //忽略其他业务代码
    
        //重写addResourceHandlers 静态资源文件映射配置
        //当每次调用registry.addResourceHandler()时,实际上它会创建一个ResourceHandlerRegistration,
        //然后将该对象放到自己的注册表管理起来,函数参数pathPatterns 表示要映射到URL pattern, 可以传递
        //多个要映射到的URL pattern。
        // pathPatterns 表示要映射到URL pattern, 可以传递多个要映射到的URLpattern,
        //addResourceLocations(...)中的可变参数则代表着对应的静态资源路径
        
        @Override
        public void addResourceHandlers(ResourceHandlerRegistry registry) {
    
            registry.addResourceHandler("swagger-ui.html")
                    .addResourceLocations("classpath:/META-INF/resources/");
            registry.addResourceHandler("/webjars/**")
                    .addResourceLocations("classpath:/META-INF/resources/webjars/");
        }
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19

    因此解决办法就是,添加默认的资源文件访问路径

        @Override
        public void addResourceHandlers(ResourceHandlerRegistry registry) {
    
            //添加默认的静态资源访问路径
           registry.addResourceHandler("/**")
                   .addResourceLocations("classpath:static/","classpath:META-IFA/resources/","classpath:resources/","classpath:public/","classpath:/");
            registry.addResourceHandler("swagger-ui.html")
                    .addResourceLocations("classpath:/META-INF/resources/");
            registry.addResourceHandler("/webjars/**")
                    .addResourceLocations("classpath:/META-INF/resources/webjars/");
    
        }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12

    新增上述代码后,访问成功!

    后记:上述调试大概花费了一个多小时,才知道原来访问默认的静态文件夹下的文件生效的原因。顺便也加深了DispatchServelet 的工作流程,看样子还是得知其然才行

    先自我介绍一下,小编13年上师交大毕业,曾经在小公司待过,去过华为OPPO等大厂,18年进入阿里,直到现在。深知大多数初中级java工程师,想要升技能,往往是需要自己摸索成长或是报班学习,但对于培训机构动则近万元的学费,着实压力不小。自己不成体系的自学效率很低又漫长,而且容易碰到天花板技术停止不前。因此我收集了一份《java开发全套学习资料》送给大家,初衷也很简单,就是希望帮助到想自学又不知道该从何学起的朋友,同时减轻大家的负担。添加下方名片,即可获取全套学习资料哦

  • 相关阅读:
    【Rust日报】2023-10-22 Korvin - 一个 WASM 前端框架,比基线 vanillajs 实现快了 33%!...
    对非均匀采样信号进行重采样
    一些知识汇总
    手机机型响应式设置2
    聊一聊Rust的enum
    基于最近电平逼近的开环MMC逆变器Simulink仿真模型
    基于armv8的kvm实现分析(三)kvm初始化流程
    关于 LLM 和知识图谱、图数据库,大家都关注哪些问题呢?
    EtherNet/IP转profienrt协议网关连接EtherNet/IP协议的川崎机器人配置方法
    java springboot2.7 写一个本地 pdf 预览的接口
  • 原文地址:https://blog.csdn.net/m0_67390969/article/details/126115453