springbean

Spring中Bean的实例化过程图示:

 

  1. public class MyBeanPostProcessor implements BeanPostProcessor {  
  2.    private Map<String, ImportListener> importMap = new ConcurrentHashMap<String, ImportListener>();
  3.      public MyBeanPostProcessor() {  
  4.         super();  
  5.         System.out.println("这是BeanPostProcessor实现类构造器!!");          
  6.      }  
  7.      
  8.      @Override  
  9.      //实例化、依赖注入完毕,在调用显示的初始化之前完成一些定制的初始化任务 
  10.      public Object postProcessBeforeInitialization(Object bean, String arg1)  
  11.              throws BeansException {  
  12.          System.out.println("bean处理器:bean创建之前..");  
  13.        
  14.          return bean;  
  15.      }  
  16.      @Override  
  17.     //实例化、依赖注入、初始化完毕时执行 
  18.      public Object postProcessAfterInitialization(Object bean, String arg1)  
  19.              throws BeansException {  
  20.          System.out.println("bean处理器:bean创建之后..");  
  21. if (bean instanceof ImportListener) {
  22. ImportListener importListener = (ImportListener) bean;
  23. importMap.put(importListener.getModuleName(), importListener);
  24. }
  25.          return bean;  
  26.      }  
  27.  }  

注意:

1、接口中的两个方法都要将传入的bean返回,而不能返回null,如果返回的是null那么我们通过getBean方法将得不到目标。

 

2、BeanFactory和ApplicationContext对待bean后置处理器稍有不同。ApplicationContext会自动检测在配置文件中实现了BeanPostProcessor接口的所有bean,并把它们注册为后置处理器,然后在容器创建bean的适当时候调用它,因此部署一个后置处理器同部署其他的bean并没有什么区别。而使用BeanFactory实现的时候,bean 后置处理器必须通过代码显式地去注册,在IoC容器继承体系中的ConfigurableBeanFactory接口中定义了注册方法:

 

在spring中,从BeanFactory或ApplicationContext取得的实例为Singleton,也就是预设为每一个Bean的别名只能维持一个实例,而不是每次都产生一个新的对象使用Singleton模式产生单一实例,对单线程的程序说并不会有什么问题,但对于多线程的程序,就必须注意安全(Thread-safe)的议题,防止多个线程同时存取共享资源所引发的数据不同步问题。
    然而在spring中 可以设定每次从BeanFactory或ApplicationContext指定别名并取得Bean时都产生一个新的实例:例如:
    在spring中,singleton属性默认是true,只有设定为false,则每次指定别名取得的Bean时都会产生一个新的实例
    一个Bean从创建到销毁,如果是用BeanFactory来生成,管理Bean的话,会经历几个执行阶段(如图1.1):
 
    1:Bean的建立:
    容器寻找Bean的定义信息并将其实例化。
    2:属性注入:
    使用依赖注入,Spring按照Bean定义信息配置Bean所有属性
    3:BeanNameAware的setBeanName():
    如果Bean类有实现org.springframework.beans.BeanNameAware接口,工厂调用Bean的setBeanName()方法传递Bean的ID。

    4:BeanFactoryAware的setBeanFactory():
    如果Bean类有实现org.springframework.beans.factory.BeanFactoryAware接口,工厂调用setBeanFactory()方法传入工厂自身。
    5:BeanPostProcessors的ProcessBeforeInitialization()
    如果有org.springframework.beans.factory.config.BeanPostProcessors和Bean关联,那么其postProcessBeforeInitialization()方法将被将被调用。
    6:initializingBean的afterPropertiesSet():
    如果Bean类已实现org.springframework.beans.factory.InitializingBean接口,则执行他的afterProPertiesSet()方法
    7:Bean定义文件中定义init-method:
    可以在Bean定义文件中使用"init-method"属性设定方法名称例如:
    如果有以上设置的话,则执行到这个阶段,就会执行initBean()方法
    8:BeanPostProcessors的ProcessaAfterInitialization()
    如果有任何的BeanPostProcessors实例与Bean实例关联,则执行BeanPostProcessors实例的ProcessaAfterInitialization()方法
    此时,Bean已经可以被应用系统使用,并且将保留在BeanFactory中知道它不在被使用。有两种方法可以将其从BeanFactory中删除掉(如图1.2):
 
    1:DisposableBean的destroy()
    在容器关闭时,如果Bean类有实现org.springframework.beans.factory.DisposableBean接口,则执行他的destroy()方法
    2:Bean定义文件中定义destroy-method
    在容器关闭时,可以在Bean定义文件中使用"destroy-method"属性设定方法名称,例如:
    如果有以上设定的话,则进行至这个阶段时,就会执行destroy()方法,如果是使用ApplicationContext来生成并管理Bean的话则稍有不同,使用ApplicationContext来生成及管理Bean实例的话,在执行BeanFactoryAware的setBeanFactory()阶段后,若Bean类上有实现org.springframework.context.ApplicationContextAware接口,则执行其setApplicationContext()方法,接着才执行BeanPostProcessors的ProcessBeforeInitialization()及之后的流程。
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值