Spring中修改allowBeanDefinitionOverriding和allowCircularReferences属性的两种方式

allowBeanDefinitionOverriding属性含义

设置是否允许通过注册具有相同名称的不同定义来覆盖bean定义,并自动替换前者。否则,将引发异常。默认值为“true”。

allowCircularReferences属性含义

设置是否允许bean之间的循环引用并自动尝试解析它们。
默认值为“true”。关闭此选项可在遇到循环引用时引发异常,完全不允许循环引用。

循环依赖应该都比较了解,主要看看allowBeanDefinitionOverriding。

public class TestSpring {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean1.xml", "bean2.xml");
        User user = (User) applicationContext.getBean("user");
        System.out.println("name: " + user.getName());
    }
}

加载了两个配置文件,并且都配置了id为user的bean对象。

bean1.xml中,name为lisi

<bean id="user" class="com.wyl.learn.User">
        <property name="name" value="lisi"></property>
        <property name="age" value="18"></property>
    </bean>

bean2.xml中,name为wangwu

<bean id="user" class="com.wyl.learn.User">
        <property name="name" value="wangwu"></property>
        <property name="age" value="18"></property>
    </bean>

根据allowBeanDefinitionOverriding属性的定义,默认值为true,并且相同名称的bean会进行覆盖,所以按照xml文件的加载顺序,bean2会覆盖bean1的配置。

最终输出结果为:name: wangwu

如果交换配置文件的加载顺序

ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean2.xml", "bean.xml");

最终输出结果为:name: lisi

如何修改属性值

方式一:

自定义一个applicationContext,并继承ClassPathXmlApplicationContext,重写customizeBeanFactory方法。

public class MyClassPathXmlApplicationContext extends ClassPathXmlApplicationContext {

    public MyClassPathXmlApplicationContext(String... configLocations) {
        super(configLocations);
    }

    @Override
    protected void customizeBeanFactory(DefaultListableBeanFactory beanFactory) {
        super.setAllowBeanDefinitionOverriding(false);
        super.setAllowCircularReferences(false);
        super.customizeBeanFactory(beanFactory);
    }

}
public class TestSpring {
    public static void main(String[] args) {
        MyClassPathXmlApplicationContext applicationContext = new MyClassPathXmlApplicationContext("bean2.xml", "bean1.xml");
        User user = (User) applicationContext.getBean("user");
        System.out.println("name: " + user.getName());
    }
}

报错如下,不允许出现定义同名的bean了。

Exception in thread "main" org.springframework.beans.factory.parsing.BeanDefinitionParsingException: Configuration problem: Failed to register bean definition with name 'user'
Offending resource: class path resource [bean1.xml]; nested exception is org.springframework.beans.factory.support.BeanDefinitionOverrideException: Invalid bean definition with name 'user' defined in class path resource [bean1.xml]: Cannot register bean definition [Generic bean: class [com.wyl.learn.User]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [bean1.xml]] for bean 'user': There is already [Generic bean: class [com.wyl.learn.User]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [bean2.xml]] bound.
	at org.springframework.beans.factory.parsing.FailFastProblemReporter.error(FailFastProblemReporter.java:72)
	at org.springframework.beans.factory.parsing.ReaderContext.error(ReaderContext.java:119)
	at org.springframework.beans.factory.parsing.ReaderContext.error(ReaderContext.java:104)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.processBeanDefinition(DefaultBeanDefinitionDocumentReader.java:314)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.parseDefaultElement(DefaultBeanDefinitionDocumentReader.java:197)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.parseBeanDefinitions(DefaultBeanDefinitionDocumentReader.java:176)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.doRegisterBeanDefinitions(DefaultBeanDefinitionDocumentReader.java:149)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.registerBeanDefinitions(DefaultBeanDefinitionDocumentReader.java:96)
	at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.registerBeanDefinitions(XmlBeanDefinitionReader.java:514)
	at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.doLoadBeanDefinitions(XmlBeanDefinitionReader.java:394)
	at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions(XmlBeanDefinitionReader.java:337)
	at org.springframework.beans.factory.xml.XmlBeanDefinitionReader.loadBeanDefinitions(XmlBeanDefinitionReader.java:305)
	at org.springframework.beans.factory.support.AbstractBeanDefinitionReader.loadBeanDefinitions(AbstractBeanDefinitionReader.java:188)
	at org.springframework.beans.factory.support.AbstractBeanDefinitionReader.loadBeanDefinitions(AbstractBeanDefinitionReader.java:224)
	at org.springframework.beans.factory.support.AbstractBeanDefinitionReader.loadBeanDefinitions(AbstractBeanDefinitionReader.java:195)
	at org.springframework.beans.factory.support.AbstractBeanDefinitionReader.loadBeanDefinitions(AbstractBeanDefinitionReader.java:257)
	at org.springframework.context.support.AbstractXmlApplicationContext.loadBeanDefinitions(AbstractXmlApplicationContext.java:128)
	at org.springframework.context.support.AbstractXmlApplicationContext.loadBeanDefinitions(AbstractXmlApplicationContext.java:94)
	at org.springframework.context.support.AbstractRefreshableApplicationContext.refreshBeanFactory(AbstractRefreshableApplicationContext.java:133)
	at org.springframework.context.support.AbstractApplicationContext.obtainFreshBeanFactory(AbstractApplicationContext.java:637)
	at org.springframework.context.support.AbstractApplicationContext.refresh(AbstractApplicationContext.java:522)
	at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:144)
	at org.springframework.context.support.ClassPathXmlApplicationContext.<init>(ClassPathXmlApplicationContext.java:95)
	at com.wyl.learn.MyClassPathXmlApplicationContext.<init>(MyClassPathXmlApplicationContext.java:9)
	at com.wyl.learn.TestSpring.main(TestSpring.java:5)
Caused by: org.springframework.beans.factory.support.BeanDefinitionOverrideException: Invalid bean definition with name 'user' defined in class path resource [bean1.xml]: Cannot register bean definition [Generic bean: class [com.wyl.learn.User]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [bean1.xml]] for bean 'user': There is already [Generic bean: class [com.wyl.learn.User]; scope=; abstract=false; lazyInit=false; autowireMode=0; dependencyCheck=0; autowireCandidate=true; primary=false; factoryBeanName=null; factoryMethodName=null; initMethodName=null; destroyMethodName=null; defined in class path resource [bean2.xml]] bound.
	at org.springframework.beans.factory.support.DefaultListableBeanFactory.registerBeanDefinition(DefaultListableBeanFactory.java:927)
	at org.springframework.beans.factory.support.BeanDefinitionReaderUtils.registerBeanDefinition(BeanDefinitionReaderUtils.java:166)
	at org.springframework.beans.factory.xml.DefaultBeanDefinitionDocumentReader.processBeanDefinition(DefaultBeanDefinitionDocumentReader.java:311)
	... 21 more

方式二:

直接调用set方法,并重新调用refresh()方法让属性生效,直接会出出现和方式一同样的报错信息。

public class TestSpring {
    public static void main(String[] args) {
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean2.xml", "bean1.xml");
        applicationContext.setAllowBeanDefinitionOverriding(false);
        applicationContext.setAllowCircularReferences(false);
        applicationContext.refresh();
        User user = (User) applicationContext.getBean("user");
        System.out.println("name: " + user.getName());
    }
}

原理分析

两种方式的关键都在于customizeBeanFactory方法

	protected void customizeBeanFactory(DefaultListableBeanFactory beanFactory) {
		if (this.allowBeanDefinitionOverriding != null) {
			beanFactory.setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);
		}
		if (this.allowCircularReferences != null) {
			beanFactory.setAllowCircularReferences(this.allowCircularReferences);
		}
	}

spirng在加载过程中,会执行到这段代码,首先判断allowBeanDefinitionOverriding属性是否为null,而这两个属性默认情况下是为null。(包装类型定义的,所以默认为null),所以beanFactory.setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);就不会被执行,
那么在DefaultListableBeanFactory中这两个属性的地方就会取默认值true。(真正使用时是通过DefaultListableBeanFactory类中定义的这个两个属性值获取的)

AbstractRefreshableApplicationContext提供了修改属性的方法,如果重写了方法,那么修改的是DefaultListableBeanFactory中的属性。

在这里插入图片描述

DefaultListableBeanFactory中的属性,默认为true。

在这里插入图片描述

所以第二种方式的原理就是,先通过set方式,让两个属性值不为null,那么就会修改DefaultListableBeanFactory中的属性了。

	protected void customizeBeanFactory(DefaultListableBeanFactory beanFactory) {
		//不为null,成立,进入if条件修改beanFactory.setAllowBeanDefinitionOverriding(true)
		if (this.allowBeanDefinitionOverriding != null) {
			beanFactory.setAllowBeanDefinitionOverriding(this.allowBeanDefinitionOverriding);
		}
		if (this.allowCircularReferences != null) {
			beanFactory.setAllowCircularReferences(this.allowCircularReferences);
		}
	}

而第一种方式避免了重新调用refresh的过程,是直接在customizeBeanFactory中进行属性赋值,赋值完成后再调用父类的customizeBeanFactory方法继续完成之后的工作,有点类似静态代理模式的运用,重写父类方法,完成自己的业务逻辑之后再调用父类原本的方法。

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

码拉松

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值