Spring系列第9篇:depend-on到底是干什么的?

public static class Bean2 implements DisposableBean {

public Bean2() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean3 implements DisposableBean {

public Bean3() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

}

上面代码中使用到了DisposableBean接口,这个是spring容器提供的一个接口,这个接口中有个destroy方法,我们的bean类可以实现这个接口,当我们调用容器的close方法关闭容器的时候,spring会调用容器中所有bean的destory方法,用来做一些清理的工作,这个以后还会细讲的。

上面几个类中构造方法和destory方法中都有输出。

下面我们来搞个测试用例看一下spring容器启动和关闭的过程中,定义的3个bean的创建和销毁的顺序。

package com.javacode2018.lesson001.demo7;

import com.javacode2018.lesson001.demo5.IocUtils;

import org.junit.Test;

import org.springframework.context.support.ClassPathXmlApplicationContext;

/**

* 公众号:路人甲Java,工作10年的前阿里P7分享Java、算法、数据库方面的技术干货!坚信用技术改变命运,让家人过上更体面的生活!

* dependon详解

*/

public class DependOnTest {

/**

* 无依赖的bean创建和销毁的顺序

*/

@Test

public void normalBean() {

System.out.println(“容器启动中!”);

String beanXml = “classpath:/com/javacode2018/lesson001/demo7/normalBean.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(“容器启动完毕,准备关闭spring容器!”);

//关闭容器

context.close();

System.out.println(“spring容器已关闭!”);

}

}

运行上面的normalBean方法,输出:

容器启动中!

class com.javacode2018.lesson001.demo7.NormalBean$Bean3 constructor!

class com.javacode2018.lesson001.demo7.NormalBean$Bean2 constructor!

class com.javacode2018.lesson001.demo7.NormalBean$Bean1 constructor!

容器启动完毕,准备关闭spring容器!

class com.javacode2018.lesson001.demo7.NormalBean$Bean1 destroy()

class com.javacode2018.lesson001.demo7.NormalBean$Bean2 destroy()

class com.javacode2018.lesson001.demo7.NormalBean$Bean3 destroy()

spring容器已关闭!

bean的定义结合上面输出我们来对比一下:

| bean定义顺序 | 创建顺序 | 销毁顺序 |

| — | — | — |

| bean3 | bean3 | bean1 |

| bean2 | bean2 | bean2 |

| bean1 | bean1 | bean3 |

从输出中可以得到2点结论:

  1. bean对象的创建顺序和bean xml中定义的顺序一致

  2. bean销毁的顺序和bean xml中定义的顺序相反

通过构造器强依赖bean创建和销毁顺序

===================

我们将上面案例改造一下,通过构造函数注入的方式使bean之间产生强依赖。

package com.javacode2018.lesson001.demo7;

import org.springframework.beans.factory.DisposableBean;

/**

* 强依赖的bean创建和销毁顺序

*/

public class StrongDependenceBean {

public static class Bean1 implements DisposableBean {

public Bean1() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean2 implements DisposableBean {

private Bean1 bean1;

public Bean2(Bean1 bean1) { //@1

this.bean1 = bean1;

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean3 implements DisposableBean {

private Bean2 bean2;

public Bean3(Bean2 bean2) { //@2

this.bean2 = bean2;

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

}

代码解释:

@1:创建Bean2的时候需要传入一个bean1对象,对bean1产生了强依赖

@2:创建Bean3的时候需要传入一个bean2对象,对bean2产生了强依赖

依赖关系是:

bean3->bean2->bean1

对应的配置(strongDependenceBean.xml):

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">

 //@1

 //@2

注意上面xml中bean定义顺序是:bean3、bean2、bean1。

@1:bean3中通过构造器注入bean2

@2:bean2中通过构造器注入bean1

DependOnTest中创建一个测试用例,如下:

/**

* 强依赖的bean的创建和销毁顺序测试

*/

@Test

public void strongDependenceBean() {

System.out.println(“容器启动中!”);

String beanXml = “classpath:/com/javacode2018/lesson001/demo7/strongDependenceBean.xml”;

ClassPathXmlApplicationContext context = IocUtils.context(beanXml);

System.out.println(“容器启动完毕,准备关闭spring容器!”);

context.close();

System.out.println(“spring容器已关闭!”);

}

运行strongDependenceBean方法输出:

容器启动中!

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean1 constructor!

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean2 constructor!

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean3 constructor!

容器启动完毕,准备关闭spring容器!

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean3 destroy()

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean2 destroy()

class com.javacode2018.lesson001.demo7.StrongDependenceBean$Bean1 destroy()

spring容器已关闭!

bean的定义结合上面输出我们来对比一下:

| bean定义顺序 | 依赖顺序(下面依赖上面的) | 创建顺序 | 销毁顺序 |

| — | — | — | — |

| bean3 | bean1 | bean1 | bean3 |

| bean2 | bean2 | bean2 | bean2 |

| bean1 | bean3 | bean3 | bean1 |

从输出中可以得到2点结论:

  1. bean对象的创建顺序和bean依赖的顺序一致

  2. bean销毁的顺序和bean创建的顺序相反

通过depend-on干预bean创建和销毁顺序

========================

上面看到了对于无依赖的bean,通过定义的顺序确实可以干预bean的创建顺序,通过强依赖也可以干预bean的创建顺序。

那么如果xml中定义的bean特别多,而有些bean之间也没有强依赖关系,此时如果想去调整bean的创建和销毁的顺序,得去调整xml中bean的定义顺序,或者去加强依赖,这样是非常不好的,spring中可以通过depend-on来解决这些问题,在不调整bean的定义顺序和强加依赖的情况下,可以通过通过depend-on属性来设置当前bean的依赖于哪些bean,那么可以保证depend-on指定的bean在当前bean之前先创建好,销毁的时候在当前bean之后进行销毁。

depend-on使用方式:

depend-on:设置当前bean依赖的bean名称,可以指定多个,多个之间可以用”,;空格“进行分割

上面不管bean2,bean2,bean4在任何地方定义,都可以确保在bean1创建之前,会先将bean2,bean3,bean4创建好,表示bean1依赖于这3个bean,可能bean1需要用到bean2、bean3、bean4中生成的一些资源或者其他的功能等,但是又没有强制去在bean1类中通过属性定义强依赖的方式去依赖于bean2、bean3、bean4;当然销毁的时候也会先销毁当前bean,再去销毁被依赖的bean,即先销毁bean1,再去销毁depend-on指定的bean。

下面我们来个案例看一下:

dependOnBean.xml

<?xml version="1.0" encoding="UTF-8"?>

<beans xmlns=“http://www.springframework.org/schema/beans”

xmlns:xsi=“http://www.w3.org/2001/XMLSchema-instance”

xsi:schemaLocation="http://www.springframework.org/schema/beans

http://www.springframework.org/schema/beans/spring-beans-4.3.xsd">

上面xml中先定义的bean3,然后定义了bean2和bean1,并且指定了bean3的depend-on=“bean2,bean1”,根据depend-on的规则,所以会先创建bean2和bean1,然后再创建bean3,销毁的时候,会按照和创建相反的顺序来,即:bean1、bean2、bean3,下面我们来看看效果是不是这样:

上面xml对应的java代码如下:

package com.javacode2018.lesson001.demo7;

import org.springframework.beans.factory.DisposableBean;

/**

* 通过depend-on来干预bean创建和销毁顺序

*/

public class DependOnBean {

public static class Bean1 implements DisposableBean {

public Bean1() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean2 implements DisposableBean {

public Bean2() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean3 implements DisposableBean {

public Bean3() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

}

DependOnTest中创建测试用例:

/**

* 通过depend-on来干预bean创建和销毁顺序

*/

@Test

读者福利

分享一份自己整理好的Java面试手册,还有一些面试题pdf

不要停下自己学习的脚步

字节跳动的面试分享,为了拿下这个offer鬼知道我经历了什么

字节跳动的面试分享,为了拿下这个offer鬼知道我经历了什么

" constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

public static class Bean3 implements DisposableBean {

public Bean3() {

System.out.println(this.getClass() + " constructor!");

}

@Override

public void destroy() throws Exception {

System.out.println(this.getClass() + " destroy()");

}

}

}

DependOnTest中创建测试用例:

/**

* 通过depend-on来干预bean创建和销毁顺序

*/

@Test

读者福利

分享一份自己整理好的Java面试手册,还有一些面试题pdf

不要停下自己学习的脚步

[外链图片转存中…(img-sZXIhZFd-1718779045331)]

[外链图片转存中…(img-AAQurL7d-1718779045332)]

  • 10
    点赞
  • 12
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值