springboot静态资源配置原理

springboot对静态资源的配置可参考WebMvcAutoConfiguration这个类,在spring-boot-autoconfigure.jar中

WebMvcAutoConfiguration

@Configuration(proxyBeanMethods = false)
@ConditionalOnWebApplication(type = Type.SERVLET)
@ConditionalOnClass({ Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class })
@ConditionalOnMissingBean(WebMvcConfigurationSupport.class)
@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE + 10)
@AutoConfigureAfter({ DispatcherServletAutoConfiguration.class, TaskExecutionAutoConfiguration.class,
		ValidationAutoConfiguration.class })
public class WebMvcAutoConfiguration {
....
}

@Configuration(proxyBeanMethods = false):告诉容器这是一个配置类,且不代理bean方法,每次调用都会创建,这样容器启动时无需检查容器中是否已经存在

@ConditionalOnWebApplication(type = Type.SERVLET):判断当前webAplication的类型是否为servlet,如果是则继续执行

@ConditionalOnClass({ Servlet.class, DispatcherServlet.class, WebMvcConfigurer.class }):判断必须存在指定的类才执行

@ConditionalOnMissingBean(WebMvcConfigurationSupport.class):判断必须没有指定的类才执行

@AutoConfigureOrder(Ordered.HIGHEST_PRECEDENCE + 10):控制执行顺序
@AutoConfigureAfter({ DispatcherServletAutoConfiguration.class, TaskExecutionAutoConfiguration.class,
ValidationAutoConfiguration.class }):在加载配置的类之后再加载当前类

WebMvcAutoConfigurationAdapter

在WebMvcAutoConfiguration这个类中,存在一个WebMvcAutoConfigurationAdapter 静态内部类,用于配置静态资源规则的。

// Defined as a nested config to ensure WebMvcConfigurer is not read when not
	// on the classpath
	@Configuration(proxyBeanMethods = false)
	@Import(EnableWebMvcConfiguration.class)
	@EnableConfigurationProperties({ WebMvcProperties.class, ResourceProperties.class })
	@Order(0)
	public static class WebMvcAutoConfigurationAdapter implements WebMvcConfigurer {
 
....

}

 这个静态内部类中有且只有一个带参构造方法:

public WebMvcAutoConfigurationAdapter(ResourceProperties resourceProperties, WebMvcProperties mvcProperties,
				ListableBeanFactory beanFactory, ObjectProvider<HttpMessageConverters> messageConvertersProvider,
				ObjectProvider<ResourceHandlerRegistrationCustomizer> resourceHandlerRegistrationCustomizerProvider) {
			this.resourceProperties = resourceProperties;
			this.mvcProperties = mvcProperties;
			this.beanFactory = beanFactory;
			this.messageConvertersProvider = messageConvertersProvider;
			this.resourceHandlerRegistrationCustomizer = resourceHandlerRegistrationCustomizerProvider.getIfAvailable();
		}

  只有一个带参构造方法时,入参会从容器中进行查找。其中的 resourceProperties和mvcProperties是通过注解@EnableConfigurationProperties导入的,其中的XXXProperties又是通过指定前缀和配置文件进行绑定的;

WebMvcProperties
@ConfigurationProperties(prefix = "spring.mvc")
public class WebMvcProperties {
ResourceProperties
@ConfigurationProperties(prefix = "spring.resources", ignoreUnknownFields = false)
public class ResourceProperties {

	private static final String[] CLASSPATH_RESOURCE_LOCATIONS = { "classpath:/META-INF/resources/",
			"classpath:/resources/", "classpath:/static/", "classpath:/public/" };

addResourceHandlers

在这个WebMvcAutoConfigurationAdapter 静态内部类中存在一个addResourceHandlers方法,这个方法就是用于静态资源规则配置使用的;

@Override
public void addResourceHandlers(ResourceHandlerRegistry registry) {
        //检查是否开启默认资源处理,默认是true,可以通过配置文件改为false,关闭所有的静态资源配置映射
       if (!this.resourceProperties.isAddMappings()) {
          logger.debug("Default resource handling disabled");
          return;
       }
       //同时通过配置文件可以添加缓存时间,秒为单位
       Duration cachePeriod = this.resourceProperties.getCache().getPeriod();
       CacheControl cacheControl = this.resourceProperties.getCache().getCachecontrol().toHttpCacheControl();
       if (!registry.hasMappingForPattern("/webjars/**")) {
       
            //对于webjars的访问全部映射到类路径下的/META-INF/resources/webjars/
          customizeResourceHandlerRegistration(registry.addResourceHandler("/webjars/**")
                .addResourceLocations("classpath:/META-INF/resources/webjars/")
                .setCachePeriod(getSeconds(cachePeriod)).setCacheControl(cacheControl));
       }
       
       //获取静态资源的路径,默认是/**
       String staticPathPattern = this.mvcProperties.getStaticPathPattern();
       if (!registry.hasMappingForPattern(staticPathPattern)) {
       
        //对于配置的静态资源路径都会去this.resourceProperties.getStaticLocations()进行查找
          customizeResourceHandlerRegistration(registry.addResourceHandler(staticPathPattern)
                .addResourceLocations(getResourceLocations(this.resourceProperties.getStaticLocations()))
                .setCachePeriod(getSeconds(cachePeriod)).setCacheControl(cacheControl));
       }
}

  this.resourceProperties.getStaticLocations():这个就对应到ResourceProperties类中的CLASSPATH_RESOURCE_LOCATIONS;

private static final String[] CLASSPATH_RESOURCE_LOCATIONS = { "classpath:/META-INF/resources/",
			"classpath:/resources/", "classpath:/static/", "classpath:/public/" };

 

spring:
#  mvc:
#    static-path-pattern: /res/**

  resources:
    add-mappings: false   禁用所有静态资源规则

  

欢迎页的处理

欢迎页使用的welcomePageHandlerMapping方法
@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;
}

  HandlerMapping:处理器映射。保存了每一个Handler能处理哪些请求。

对于这个有参构造方法:

WelcomePageHandlerMapping(TemplateAvailabilityProviders templateAvailabilityProviders,
			ApplicationContext applicationContext, Optional<Resource> welcomePage, String staticPathPattern) {
		if (welcomePage.isPresent() && "/**".equals(staticPathPattern)) {
			logger.info("Adding welcome page: " + welcomePage.get());
			setRootViewName("forward:index.html");
		}
		else if (welcomeTemplateExists(templateAvailabilityProviders, applicationContext)) {
			logger.info("Adding welcome page template: index");
			setRootViewName("index");
		}
	}

  可以看出判断条件,如果开启了欢迎页且静态路径必须匹配/**则会去index.html,要么是跳转到index的请求。

      至此可以看出springboot对静态资源的所有默认规则。

 

posted @ 2022-07-29 16:17  swayer  阅读(366)  评论(0编辑  收藏  举报