Spring IOC篇

一、什么是控制反转?

参考这篇回答,更有助于理解。
Spring IoC有什么好处呢?-Mingqi的回答

二、为什么需要控制反转?

简单来说就是解耦。

三、IOC的原理

控制反转IOC可以通过两种方式实现,依赖查找DL,依赖注入DI。DL已经被抛弃;DI是Spring使用的方式,除了Spring以外,Google的Guice,PicoContainer等也用了这种方式实现IOC。

Spring的IOC设计支持以下功能:
1.依赖注入
2.依赖检查
3.自动装配
4.支持集合
5.指定初始化方法和销毁方法
6.支持回调某些方法(需要实现Spring接口,略有侵入性)

1.Spring IOC 容器

低级容器BeanFactory
高级容器ApplicationContext
其实实现IOC低级容器BeanFactory就够了,整个过程分两步:
1.低级容器加载配置文件(xml、数据库或者Applet),解析成BeanDefinition放到Map里;
2.加载成功后,高级容器启动高级功能(例如Bean后置处理器,回调setBeanFactory方法,注册监听,实例化单例Bean等等)。调用getBean的时候,从BeanDefinition所属的Map里,拿出Class进行实例化,如果有依赖关系,则递归调用getBean方法,完成依赖注入。

2.Bean的生命周期
Bean的生命周期
  1. ResouceLoader加载配置信息
  2. BeanDefintionReader解析配置信息,生成一个一个的BeanDefintion
  3. BeanDefintion由BeanDefintionRegistry管理起来
  4. BeanFactoryPostProcessor对配置信息进行加工(也就是处理配置的信息,一般通过PropertyPlaceholderConfigurer来实现)
  5. 实例化Bean
  6. 如果该Bean配置/实现了InstantiationAwareBean,则调用对应的方法
  7. 使用BeanWarpper来完成对象之间的属性配置(依赖)
  8. 如果该Bean配置/实现了Aware接口,则调用对应的方法
  9. 如果该Bean配置了BeanPostProcessor的before方法,则调用
  10. 如果该Bean配置了init-method或者实现InstantiationBean,则调用对应的方法
  11. 如果该Bean配置了BeanPostProcessor的after方法,则调用
  12. 将对象放入到HashMap中
  13. 最后如果配置了destroy或者DisposableBean的方法,则执行销毁操作
3.Bean的作用域

我们可使用<bean>标签的scope属性来指定一个Bean的作用域,如下:



1)singleton

Spring默认的作用域,在这样的作用域下,每一个Bean的实例只会被创建一次,而且Spring容器在整个应用程序生存期中都可以使用该实例。因此之前的代码中spring容器创建Bean后,通过代码获取的bean,无论多少次,都是同一个Bean的实例。

2)prototype

除了Singleton外还有另外一种比较常用的作用域,prototype,它代表每次获取Bean实例时都会新创建一个实例对象,类似new操作符。

3)request 和 session

在spring2.5中专门针对Web应该程序引进了request和session这两种作用域。从名称不难看出,request作用域的bean实例,仅在当前HTTP request内有效,不同的request获取到的是不同的bean实例;session作用域的bean实例,仅在当前HTTP session内有效,不同的session获取到的是不同的bean实例。
需要注意的是,声明prototype, request, session三种作用域时,还需要声明代理模式。
例如:

@Component
@Scope(value = "singleton")
public class SingletonBean {
    //......
}

@Component
@Scope(value = "prototype" , proxyMode = ScopedProxyMode.TARGET_CLASS)
public class PrototypeBean {
    //......
}

@Component
@Scope(value = "request" , proxyMode = ScopedProxyMode.TARGET_CLASS)
public class RequestBean {
    //......
}

@Component
@Scope(value = "session" , proxyMode = ScopedProxyMode.TARGET_CLASS)
public class SessionBean {
    //........
}

还有





    
    



    
    


还需要注意,这种经过xml声明代理的方式不适合prototype作用域,该作用域生效的方式目前测试中只有基于注解方式和基于实现ApplicationContextAware接口两种方式。

4)globalSession作用域

这种作用域类似于Session作用域,相当于全局变量,类似Servlet的Application,适用基于portlet的web应用程序,请注意,portlet在这指的是分布式开发,而不是portlet语言开发。

你可能感兴趣的:(Spring IOC篇)