ResourceLoader详解(三)--DefaultReso

2019-03-19  本文已影响0人  北海北_6dc3

对象

ResourceLoader继承对象.png

DefaultResourceLoader目的

{@link ResourceLoader}接口的默认实现。由{@link ResourceEditor}使用,并作为{@link org.springframework.context.support.AbstractApplicationContext}的基类。也可以单独使用。

如果位置值是URL,则返回{@link UrlResource};如果是非URL路径或“classpath:”伪URL,则返回{@link ClassPathResource}。

构造函数

    //设置默认ClassLoder
    public DefaultResourceLoader() {
        this.classLoader = ClassUtils.getDefaultClassLoader();
    }

    public DefaultResourceLoader(@Nullable ClassLoader classLoader) {
        this.classLoader = classLoader;
    }
    //..setClassLoader、getClassLoader方法

我们可以看到构造函数的主要目的就是赋值classLoader

protocolResolvers

    //添加protocolResolvers
    public void addProtocolResolver(ProtocolResolver resolver) {
        Assert.notNull(resolver, "ProtocolResolver must not be null");
        this.protocolResolvers.add(resolver);
    }

    //获取
    public Collection<ProtocolResolver> getProtocolResolvers() {
        return this.protocolResolvers;
    }
    //调用核心方法
    @Override
    public Resource getResource(String location) {
        //我们看到,相当于一个拦截器,优先执行查找resource,找到即返回
        for (ProtocolResolver protocolResolver : this.protocolResolvers) {
            Resource resource = protocolResolver.resolve(location, this);
            if (resource != null) {
                return resource;
            }
        }
        //后续代码,不考虑拦截流程。
        if (location.startsWith("/")) {
            return getResourceByPath(location);
        }
        else if (location.startsWith(CLASSPATH_URL_PREFIX)) {
            return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());
        }
        else {
            try {
                // Try to parse the location as a URL...
                URL url = new URL(location);
                return (ResourceUtils.isFileURL(url) ? new FileUrlResource(url) : new UrlResource(url));
            }
            catch (MalformedURLException ex) {
                // No URL -> resolve as resource path.
                return getResourceByPath(location);
            }
        }
    }

我们可以看到protocolResolvers,相当于对默认实现的一个增强,传入其他解析查找模式,如果匹配则返回。
默认查找方式主要分为三种
1.匹配开头斜杠和异常。getResourceByPath
2.匹配格式classpath:。ClassPathResource
3.尝试用FileUrlResource或者UrlResource获取
首先我们来分析第一种情况

getResourceByPath

    protected Resource getResourceByPath(String path) {
        return new ClassPathContextResource(path, getClassLoader());
    }
    /**
     * 通过实现ContextResource接口显式表达上下文相关路径的ClassPathResource,此方法重点在于实现ContextResource
     */
    protected static class ClassPathContextResource extends ClassPathResource implements ContextResource {
        //基于父类构造
        public ClassPathContextResource(String path, @Nullable ClassLoader classLoader) {
            super(path, classLoader);
        }

        @Override
        public String getPathWithinContext() {
            return getPath();
        }
        //基于当前路径的重新构建对象
        @Override
        public Resource createRelative(String relativePath) {
            String pathToUse = StringUtils.applyRelativePath(getPath(), relativePath);
            return new ClassPathContextResource(pathToUse, getClassLoader());
        }
    }

我们可以看到getResourceByPath,可以简单的看做对ClassPathResource 的增强。即调用ClassPathResource 进行解析。
ClassPathResource解析,查看第五节

从上可以看出,第一种情况和第二种情况都是调用ClassPathResource 实现

总结

DefaultResourceLoader功能有2:
1.增加基于protocolResolvers的拦截增强
2.基于ClassPathResource 、FileUrlResource、UrlResource解析,优先级从左到右

上一篇 下一篇

猜你喜欢

热点阅读