Spring IOC 之统一资源加载策略

统一资源:Resource
 

  • org.springframework.core.io.Resource 为 Spring 框架所有资源的抽象和访问接口
    • 它继承 org.springframework.core.io.InputStreamSource接口
    • 作为所有资源的统一抽象,Source 定义了一些通用的方法,由子类 AbstractResource 提供统一的默认实现

3f8af9f69cb444cdc450f273794b0e97d55.jpg

  • FileSystemResource:对 java.io.File 类型资源的封装,只要是跟 File 打交道的,基本上与 FileSystemResource 也可以打交道。支持文件和 URL 的形式,实现 WritableResource 接口,且从 Spring Framework 5.0 开始,FileSystemResource 使用NIO.2 API进行读/写交互
  • ByteArrayResource:对字节数组提供的数据的封装。如果通过 InputStream 形式访问该类型的资源,该实现会根据字节数组的数据构造一个相应的 ByteArrayInputStream。
  • UrlResource:对 java.net.URL类型资源的封装。内部委派 URL 进行具体的资源操作。
  • ClassPathResource:class path 类型资源的实现。使用给定的 ClassLoader 或者给定的 Class 来加载资源。
  • InputStreamResource:将给定的 InputStream 作为一种资源的 Resource 的实现类。  

统一资源定位:ResourceLoader

  • org.springframework.core.io.ResourceLoader 为 Spring 资源加载的统一抽象,具体的资源加载则由相应的实现类来完成
  • 所以我们可以将 ResourceLoader 称作为统一资源定位器
  • 该方法的主要实现是在其子类 DefaultResourceLoader 中实现
  • ResourceLoader 接口提供两个方法:getResource()getClassLoader()
  • getResource() 方法支持以下模式的资源加载:
    • URL位置资源,如”file:C:/test.dat”
    • ClassPath位置资源,如”classpath:test.dat”
    • 相对路径资源,如”WEB-INF/test.dat”,此时返回的Resource实例根据实现不同而不同

2fe5aa1af49c8a3a898a89d35fd843e7ab1.jpg

DefaultResourceLoader

  • DefaultResourceLoader 是 ResourceLoader 的默认实现
  • 首先通过 ProtocolResolver 来加载资源,成功返回 Resource,否则调用如下逻辑:
    • 若 location 以 / 开头,则调用 getResourceByPath()构造 ClassPathContextResource 类型资源并返回
    • 若 location 以 classpath: 开头,则构造 ClassPathResource 类型资源并返回,在构造该资源时,通过 getClassLoader()获取当前的 ClassLoader
    • 构造 URL ,尝试通过它进行资源定位,若没有抛出 MalformedURLException 异常,则判断是否为 FileURL , 如果是则构造 FileUrlResource 类型资源,否则构造 UrlResource。若在加载过程中抛出 MalformedURLException 异常,则委派 getResourceByPath() 实现资源定位加载
  • ProtocolResolver ,用户自定义协议资源解决策略,作为 DefaultResourceLoader 的 SPI,它允许用户自定义资源加载协议,而不需要继承 ResourceLoader 的子类

FileSystemResourceLoader

  • FileSystemContextResource 为 FileSystemResourceLoader 的内部类,它继承 FileSystemResource

ResourcePatternResolver

  • ResourcePatternResolver 是 ResourceLoader 的扩展,它支持根据指定的资源路径匹配模式每次返回多个 Resource 实例
  • ResourcePatternResolver 新增的 classpath*: 前缀外,还支持 Ant 风格的路径匹配模式(类似于 **/*.xml
  • PathMatchingResourcePatternResolver 在实例化的时候,可以指定一个 ResourceLoader,如果不指定的话,它会在内部构造一个 DefaultResourceLoader

1b674b8ec468f8b26131693a7e494374414.jpg

  • Resource[] getResources(String locationPattern) 方法处理逻辑:

cd9066a60792637d08a8031cc3758e8c64c.jpg

  • protected Resource[] findPathMatchingResources(String locationPattern)
  • 主要分两步:

    • 确定目录,获取该目录下得所有资源

    • 在所获得的所有资源中进行迭代匹配获取我们想要的资源。

我们要关注两个方法,一个是 determineRootDir(),一个是 doFindPathMatchingFileResources()

  • determineRootDir()主要是用于确定根路径

b6034e055af40d2350b19b30f18db2f2665.jpg

下面简要总结下:

  • Spring 提供了 Resource 和 ResourceLoader 来统一抽象整个资源及其定位。使得资源与资源的定位有了一个更加清晰的界限,并且提供了合适的 Default 类,使得自定义实现更加方便和清晰
  • DefaultResource 为 Resource 的默认实现,它对 Resource 接口做了一个统一的实现,子类继承该类后只需要覆盖相应的方法即可,同时对于自定义的 Resource 我们也是继承该类
  • DefaultResourceLoader 同样也是 ResourceLoader 的默认实现,在自定 ResourceLoader 的时候我们除了可以继承该类外还可以实现 ProtocolResolver 接口来实现自定资源加载协议
  • DefaultResourceLoader 每次只能返回单一的资源,所以 Spring 针对这个提供了另外一个接口 ResourcePatternResolver ,该接口提供了根据指定的 locationPattern 返回多个资源的策略。其子类 PathMatchingResourcePatternResolver 是一个集大成者的 ResourceLoader ,因为它即实现了 Resource getResource(String location) 也实现了 Resource[] getResources(String locationPattern)

转载于:https://my.oschina.net/u/3847203/blog/2249222

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值