Spring核心——Bean的定义与控制

在前面两篇介绍Sring核心与设计模式的文章中,分别介绍了Ioc容器和Bean的依赖关系。如果阅读过前2文就会知道,Spring的整个运转机制就是围绕着IoC容器以及Bean展开的。IoC就是一个篮子,所有的Bean都向里面扔。除了提供篮子功能创建并存放Bean之外,IoC还要负责管理Bean与Bean之间的关系——依赖注入。之前也提到Bean是Spring核心容器的最小工作单元,Spring一些更高级的功能(例如切面、代理)都是在Bean的基础上实现。

除了管理Bean与Bean之间的关系,IoC还提供了对Bean自身进行控制的各项功能,本文将介绍Bean的生命周期功能以及状态定义功能。

前置依赖
Bean与Bean之间存在依赖关系,可以是强依赖(通过XML和注解直接声明依赖)、也可以是弱依赖(ApplicationContextAware等方式获取)。当一个Bean需要另外一个Bean完成初始化后自身才能工作时,例如一个Bean依赖DataSoruce,但是DataSource的初始化需要较长时间。这个时候用depends-on声明前置依赖即可:

延迟加载
通常情况下,所有的 singleton 类型的Bean都会在容器创建后进行初始化,简单的说就是启动Jvm就开始创建(实际上是创建ApplicationContext的某个实现类实例之后)。

IoC支持所有的 singleton Bean在使用时再加载,这样做的好处是可以大大节省初始化的时间。但是如果你的应用对启动时间的长短并不敏感,建议让所有的 singleton 都启动时加载。这样可以在启动时就发现一些问题,而不是在运行很久直到使用时才由用户去触发这个问题。或者可以根据场景来使用决定是否延迟,例如开发时使用延迟加载,而在集成测试或上生产时关闭。

可以设置全局延迟加载,也可以设置某个Bean延迟加载:

需要注意的是,在设置某个单独的Bean延迟加载时,如果有某个没有延迟加载的Bean要依赖他,那实际上也会在初始化的时候就加载。

还要强调一下,这里的“加载”仅仅是为了表示一个类被Ioc创造并放置容器中,和classLoad方法将class文件中的字节码加载到方法区的加载是两个概念。

延迟加载在设计模式上是单例模式一种延伸,通常也被称为懒汉模式。单例通常有双重锁+volatile、静态类和枚举三种方式实现。在Effective Java一书中对三种模式都有深入的解析。而对于Spring容器而言,枚举的方式肯定不好用了,静态类由于属于自身代码级别应该也不会用,所以双重锁的实现方式较为可信。不过我没去看过源码,仅属于猜测。

生命周期方法
一个Bean的创建、使用再到最后销毁称为"Bean的生命周期"。Spring框架为Bean的生命周期各个阶段提供了多种回掉方法来处理各种状态或者数据。

初始化方法

当一个Bean完成初始化并注入各项参数之后,初始化回掉方法会被调用,简单的说就是完成创建之后会被调用。实现初始化回调方法有2个路径:1.继承org.springframework.beans.factory.InitializingBean接口,然后实现 afterPropertiesSet方法。2.在Bean的XML配置上使用init-method属性来制定要调用的初始化:

继承实现:

配置实现:

2种方法都等效,实际使用是我们应该使用哪一种方法呢?

InitializingBean是Spring早期实现的一个生命周期回调方法。但是在JCP推出JSR-250和JSR-330规范之后,Spring的大神们开始意识到基于元编程思想和配置手段来实现非侵入式框架(Not Coupled)才是正道。所以现在都是推荐使用配置文件和JSR-250的@PostConstruct(关于各种Annotation的使用请关注后续的文章)。现在依然保留InitializingBean应该是考虑到兼容问题。

销毁方法
与创建方法相对应的是销毁方法。当一个类将要被销毁之前,对应的销毁回调方法会被调用。销毁方法也有一个继承实现和配置+注解实现:

继承实现:

配置实现:


依然建议销毁手段也使用配置或@PreDestroy来设定销毁方法。

全局配置初始化与销毁方法
IoC容器还提供了全局配置初始化与销毁方法的配置:

通过在<beans>标签上使用default-init-method和default-destroy-method 属性参数,可以为容器中所有的Bean统一指定初始化和销毁的生命周期方法。

如果在<beans>上设定2个默认的生命周期方法,同时在<bean>上也指定了init-method或destroy-method,回调方法会以<bean>上的配置为准。这样就保证全局配置与单独配置可以共存。

使用初始化或销毁2个生命周期方法注意的要点:

初始化和销毁都提供了3种手段:XML配置、注解、以及实现接口。系统的各个部分会交由不同的团队开发,不遵循统一的规范,建议使用满足JSR规范的注解——@PostConstruct、@PreDestroy。如果是统一的团队,准训一致的规范,建议使用<beans>的属性统一名称使用全局配置。
如果Bean设计到代理模式时(例如使用了AOP),那么生命周期方法被调用时,有可能代理类还没有被创建出来。因为生命周期方法是实体类完成对应工作之后就会被调用,而与代理类无关。

原文地址:http://blog.51cto.com/13754022/2134166

时间: 2024-07-29 16:16:27

Spring核心——Bean的定义与控制的相关文章

spring:bean的定义

一个完整的Bean的配置文件: <?xml version="1.0" encoding="utf-8"?> <!DOCTYPE beans PUBLIC "-//SPRING//DTD//BEAN//EN" "http://www.springframework.org/dtd/spring-beans.dtd"> <beans> <!--定义一个bean,id是这个bean的唯一标

Spring核心思想:IOC(控制反转)、DI(依赖注入)和AOP(面向切面编程)

Spring有三大核心思想,分别是控制反转(IOC,Inversion Of Controller),依赖注入(DI,Dependency Injection)和面向切面编程(AOP,Aspect Oriented Programming). 控制反转(IOC,Inversion Of Controller) 控制反转不是什么技术,而是一种设计思想.在Java开发中,IOC意味着将你设计好的对象交给容器控制,而不是传统的在你的对象内部直接控制. 传统正转 控制有反转,当然也就有原来的正转.正转就

Spring进阶之路(1)-Spring核心机制:依赖注入/控制反转

我们常常会遇到这样一种情景.就是在我们开发项目的时候常常会在一个类中调用其它的类中的方法,来完毕我们期望的任务.大部分的情况下往往会採用在当前需要的这个类里面new一个实例出来.然后调用他的方法,那么这种话会有个问题.就是有一天我想改变下这个类,改为其它的名称.那么这时候必需要做的是同一时候去调用方的类文件里改变这个改变的类的名称.这种情况是由于代码的耦合带来了后期维护成本的添加,那么spring的出现就能够非常好的起到解耦的作用,而他的核心机制就是依赖注入. 依赖注入与控制反转 依赖注入:对于

Spring中Bean的定义及作用域的注解实现

Classpath扫描与组件管理: 从Spring3.0开始,Spring JavaConfig项目提供了很多特性,包括使用java而不是xml定义bean,指的是注解 @Configuration,@Bean ,@Import ,@DependsOn @Component是一个通用注解,可用于任何bean @Repository:通常用于注解DAO类,即持久层 @Service:通常用于注解Service类,即服务层 @Controller:通常用于Controller类,即控制层MVC 元注

Spring核心知识点

目录 Spring概述 依赖注入 Spring Beans Spring注解 Spring的对象访问 Spring面向切面编程 Spring MVC框架 Spring概述 1.什么是Spring? Spring是一个开源的Java EE开发框架.Spring框架的核心功能可以应用在任何Java应用程序中,但对Java EE平台上的Web应用程序有更好的扩展性.Spring框架的目标是使得Java EE应用程序的开发更加简捷,通过使用POJO为基础的编程模型促进良好的编程风格. 2.Spring有

spring中bean配置和bean注入

1 bean与spring容器的关系 Bean配置信息定义了Bean的实现及依赖关系,Spring容器根据各种形式的Bean配置信息在容器内部建立Bean定义注册表,然后根据注册表加载.实例化Bean,并建立Bean和Bean的依赖关系,最后将这些准备就绪的Bean放到Bean缓存池中,以供外层的应用程序进行调用. 1 bean配置 bean配置有三种方法: 基于xml配置Bean 使用注解定义Bean 基于java类提供Bean定义信息 1.1 基于xml配置Bean 1.2 使用注解定义Be

Spring学习九----------Bean的配置之Bean的定义及作用域的注解实现

Spring Bean常用注解 @Component:通常注解,可用于任何Bean @Repository:通常用于注解DAO层,即持久层 @Service:通常用于注解Service层,即服务层 @Controller:通常用于注解Controller层,即控制层 类的自动检测及Bean的注册 <context:component-scan base-package=""/>:自动扫描base-package定义的包或其子包下的类,并将带有@Component,@Cont

spring三大核心学习(一)---控制反转

记得当年大学时候,java的企业级框架还是ssh的天下(spring,struts和hibernate),但是现在,感觉spring已经完全把那两个框架甩在后边了.用spring的人越来越多,用struts的人比原来少多了,用hibernate的就更少了.毕竟现在it人才流动性太大了,招人上来上手很重要,谁知道他什么时候就跑了..(hibernate太复杂了). 我最近只看spring,所以只说spring. 记得当初面试的时候被问到过spring的三大核心思想.相信多数javaer都被问到过这

Spring核心学习(3)为Bean注入属性

前导:开始学习Spring核心思想,通过一个山寨精简版Spring代码结合学习. 内容:1.Propertyvalue-保存属性注入信息.2.AutowireCapableBeanFactory-可自动装配的BeanFactory. 这里我们重新定义了BeanDefinition,增加了属性列表这个字段,我们将为bean附加额外的属性,所以我们又定了PropertyValues.PropertyValue来辅助,当我们为bean定义的时候同时设置他的属性,通过反射机制来动态的附加到bean里,来