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

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

声明前置依赖即可:

<!-- 依赖多个Bean使用,号分割 -->
<bean id="beanOne" class="ExampleBean" depends-on="manager,accountDao">
    <property name="manager" ref="manager" />
</bean>

<bean id="manager" class="ManagerBean" />
<bean id="accountDao" class="x.y.jdbc.JdbcAccountDao" />复制代码

延迟加载

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

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

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

<beans default-lazy-init="true">
    <!-- 所有的Bean知道使用的时候才会进行加载... -->
</beans>复制代码
<!-- 只有lazy类延迟加载 -->
<bean id="lazy" class="com.foo.ExpensiveToCreateBean" lazy-init="true"/>
<bean name="not.lazy" class="com.foo.AnotherBean"/>复制代码

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

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

延迟加载在设计模式上是单例模式一种延伸,通常也被称为懒汉模式。单例通常有双重锁+volatile、静态类和枚举三种方式实现。在Effective

一书中对三种模式都有深入的解析。而对于Spring容器而言,枚举的方式肯定不好用了,静态类由于属于自身代码级别应该也不会用,所以双重锁的实现方式较为可信。不过我没去看过源码,仅属于猜测。

生命周期方法

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

初始化方法

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

继承实现:

<bean id="a" class="x.y.A" />复制代码
package x.y;
public class A implements InitializingBean {
    public void afterPropertiesSet(){
        // init
    }
}复制代码

配置实现:

<bean id="a" class="x.y.A" init-method="init" />复制代码
package x.y;
public class A {
    public void init(){}
}复制代码

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

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

销毁方法

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

继承实现:

<bean id="a" class="x.y.A" />复制代码
package x.y;
public class A implements DisposableBean {
    public void destroy(){
        // 销毁资源
    }
}复制代码

配置实现:

<bean id="a" class="x.y.A" destroy-method="cleanUp" />复制代码
package x.y;
public class A {
    public void cleanUp(){
        // 销毁资源
    }
}复制代码

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

全局配置初始化与销毁方法

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

package x.y;
public class A {
    public void init(){
        // 初始化资源
    }
    public void destroy(){
        // 销毁资源
    }
}复制代码
<beans default-init-method="init" default-destroy-method="destroy">
     <bean id="a" class="x.y.A"/>
     <!-- bean configuration -->
</beans>复制代码

通过在<beans>标签上使用

default-init-method

default-destroy-method

属性参数,可以为容器中所有的Bean统一指定初始化和销毁的生命周期方法。

如果在<beans>上设定2个默认的生命周期方法,同时在<bean>上也指定了

init-method

destroy-method,

回调方法会以<bean>上的配置为准。这样就保证全局配置与单独配置可以共存。

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

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

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/101587.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • wpf滑动动画_旋转平移矩阵

    wpf滑动动画_旋转平移矩阵在WPF动画中常见的动画就平移、缩放以及旋转,一般会用到故事板(Storyboard)和浮点动画(DoubleAnimation),下面我们先对这两个进行具体的介绍,因为本次我们主要利用故事板来添加动画:一.Storyboard:俗称故事板,在VS中意思是,一个容器的时间线,该时间线为子动画提供对象和属性确定信息:其重要的两个方法`Storyboard.SetTarget();`绑定动画和`Storyboard.SetTargetProperty();`依赖的属性一般用法有:Storyboard

    2025年8月25日
    3
  • 7. MyBatisCodeHelperPro 插件[通俗易懂]

    7. MyBatisCodeHelperPro 插件[通俗易懂]MyBatisCodeHelperPro插件       之前介绍了MyBatis的逆向工程生成实体类、自定义接口以及Mapper.xml的操作过程,很显然这样的配置工作非常繁琐且复杂,本文在介绍MyBatis动态SQL之前,先分享一款实用的MyBatisGenerator插件(MyBatisCodeHelperPro),使用这个插件可以简化开发过程,提高开发者的编码效率。第一步:File→Settings

    2022年9月15日
    5
  • 面试题,说说你对spring IOC和AOP的理解

    面试题,说说你对spring IOC和AOP的理解https://blog.csdn.net/s20081294/article/details/44677387在面试中,经常会问,说说你对springIOC和AOP的理解,问题很宽泛,似乎不知道从何说起。回答思路:1.先用通俗易懂的话解释下何为IOC和AOP———》2.各自的实现原理———–》3.自己的项目中如何使用以下是个人的一些总结,仅供参考。1.IOC…

    2022年6月16日
    66
  • Mac book Pro_mac安装sas

    Mac book Pro_mac安装sasMacPorts和Homebrew都是MacOSX上的软件包管理工具(viaWikipedia),且它们之间是不兼容的.个中好处就不介绍了,这里要说的是删除MacPorts并安装Homebrew.准备条件:Mac是自带Ruby程式的,如果你之间”处理”过它,记得要确保它的…

    2022年9月21日
    2
  • MicroBlaze使用_char* malloc

    MicroBlaze使用_char* malloc转自http://blog.163.com/gcs_gcs/blog/static/17448606620121193113914/在最近的工程中,需要用到PS/2键盘和鼠标作为控制输入,所以在网上找了一些相关的资料,内容很丰富,看来已经有很多人做过了这方面的编程。本篇Blog算是实践总结,为以后的开发积累一些基础知识。MicroBlaze支持重启(reset),中断(interrupt),暂…

    2025年8月18日
    6
  • macbookpro双系统分区_macbook双系统分区调整

    macbookpro双系统分区_macbook双系统分区调整一些前序知识:1、Mac系统不是Intel的i386架构,没有Bios,但是有EFI,通过EFI管理系统的引导。2、Mac系统的分区表采用GUID,不是MBR。因此如果硬盘完全交给windows控制,会导致Mac系统无法启动。3、任何对MBR的强行操作,会导致已经装好的双系统引导失效。4、Mac的EFI分区会用GPT锁定,不要尝试去操作这个分区。我们…

    2022年10月5日
    2

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号