首页 > 代码库 > BeanDefinition的Resource定位——3

BeanDefinition的Resource定位——3

1.我们重点看看AbstractRefreshableApplicationContextrefreshBeanFactory方法的实现,这个refreshBeanFactoryFileSystemXmlApplicationContext构造函数中的refresh调用。在这个方法中,通过createBeanFactory构建了一个IoC容器供ApplicationContext使用。这个IoC容器就是我们前面提到过的DefaultListableBeanFactory,同时,它启动了loadBeanDefinitions来载入BeanDefinition,这个过程和前面以编程的方式来使用IoC容器(XmlBeanFacoty)的过程非常类似。

 1 protected final void refreshBeanFactory() throws BeansException { 2         // 这里判断,如果已经建立了BeanFactory,则销毁并关闭该BeanFactory 3         if (hasBeanFactory()) { 4             destroyBeans(); 5             closeBeanFactory(); 6         } 7         // 这里是创建并设置持有的DefaultListableBeanFactory的地方 8         // 同时调用loadBeanDefinitions载入BeanDefinition的信息 9         try {10             DefaultListableBeanFactory beanFactory = createBeanFactory();11             beanFactory.setSerializationId(getId());12             customizeBeanFactory(beanFactory);13             loadBeanDefinitions(beanFactory);14             synchronized (this.beanFactoryMonitor) {15                 this.beanFactory = beanFactory;16             }17         }18         catch (IOException ex) {19             throw new ApplicationContextException("I/O error parsing bean definition source for " + getDisplayName(), ex);20         }21     }
AbstractRefreshableApplicationContext对容器的初始化

 

1 // 这就是在上下文中创建DefaultListableBeanFactory的地方,而getInternalParentBeanFacoty()2     // 的具体实现可以3     // 查看AbstractApplicationContext中的实现,会根据容器已有的双亲IoC容器的信息来生成4     // DefaultListableBeanFactory的双亲IoC容器5     protected DefaultListableBeanFactory createBeanFactory() {6         return new DefaultListableBeanFactory(getInternalParentBeanFactory());7     }
createBeanFactory

 

1 // 这里是使用BeanDefinitionReader载入Bean定义的地方,因为允许有多种载入方式,虽然用得2     // 最多的事XML定义的形式,这里通过一个抽象函数把具体的实现委托给子类来完成3     protected abstract void loadBeanDefinitions(DefaultListableBeanFactory beanFactory)4             throws BeansException, IOException;
loadBeanDefinitions
 1 // 对于取得Resource的具体过程,我们可以看看DefaultResourceLoader是怎样完成的 2     public Resource getResource(String location) { 3         Assert.notNull(location, "Location must not be null"); 4         //  5         if (location.startsWith("/")) { 6             return getResourceByPath(location); 7         } 8         else if (location.startsWith(CLASSPATH_URL_PREFIX)) { 9             return new ClassPathResource(location.substring(CLASSPATH_URL_PREFIX.length()), getClassLoader());10         }11         else {12             try {13                 // Try to parse the location as a URL...14                 URL url = new URL(location);15                 return new UrlResource(url);16             }17             catch (MalformedURLException ex) {18                 // No URL -> resolve as resource path.19                 return getResourceByPath(location);20             }21         }22     }
getResource

 

 2.前面我们看到的getResourceByPath被子类FileSystemXmlApplicationContext实现,这个方法返回的是一个FileSystemResource对象,通过这个对象,Spring可以进行相关的I/O操作,完成BeanDefinition的定位。分析到这里已经一目了然,它实现的就是对path进行解析,然后生成一个FileSystemResource对象并返回,代码清单如下所示。

 

1 protected Resource getResourceByPath(String path) {2         if (path != null && path.startsWith("/")) {3             path = path.substring(1);4         }5         return new FileSystemResource(path);6     }
FileSystemXmlApplicationContext生成FileSystemResource对象

 

3.如果是其他的ApplicationContext,那么会对应生成其他种类的Resource,比如ClassPathResource、ServletContextResource等。

4.关于Spring中Resource的种类,可以在图1的Resource类的继承关系中了解。作为接口Resource定义了许多与I/O相关的操作,这些操作也都可以从图1种的Resource的接口定义中看到。这些接口对不同的Resource实现代表着不同的意义,是Resource的实现需要考虑的。Resource接口的实现在Spring中的设计如图1所示。

图1 Resource的定义和继承关系

5.从图1中我们可以看到Resource的定位和它的继承关系,通过对前面的实现原理的分析,我们以FileSystemXmlApplicationContext的实现为例子,了解Resource定位问题的解决方案,即以FileSystem方式存在的Resource的定位实现。

6.在BeanDefinition定位完成的基础上,就可以通过返回的Resource对象来进行BeanDefinition的载入了。

7.在定位过程完成以后,为BeanDefinition的载入创造了I/O操作的条件,但是具体的数据还没有开始读入。这些数据的读入将在下面的BeanDefinition的载入和解析中来完成

8.任然以水桶为例子,这里就像用水桶去打水,要先找到水源。这里完成对Resource的定位,就类似于水源已经找到了,下面就是打水的过程了,类似于把找到的水装到水桶里的过程。找水不简单,但是与打水相比,我们发现打水更需要技巧。

 

BeanDefinition的Resource定位——3