IOC之bean之间的关系讲解

x33g5p2x  于2021-03-14 发布在 其他  
字(7.2k)|赞(0)|评价(0)|浏览(277)

本文主要讲了Spring中Bean之间的关系,分为继承、依赖和引用三个类型。文章中都分别有例子和用法分析。

一、继承Bean配置

  • Spring允许继承bean的配置 ,被继承的bean称为父bean,继承这个父Bean的Bean称为子Bean

  • 子Bean从父Bean中继承配置,包括Bean的属性配置

  • 子Bean也可以 覆盖 从父Bean继承过来的配置

  • 父Bean可以作为配置模版,也可以作为Bean实例, 若只想把父Bean作为模版,可以配置<bean>的abstract属性为true ,这样Spring将不会实例化这个Bean

  • 并不是<bean>元素里的所有属性都会被继承 。比如:autowire,abstract等。

  • 也 可以忽略父Bean的class属性 ,让子Bean指定自己的类,而共享相同的属性配置,但此时 abstract必须设为true

使用案例:

定义House.java:

package com.mucfc;

public class House {
	private String houseSize;
	private String housePosition;
	private String housePrice;
	public String getHouseSize() {
		return houseSize;
	}
	public void setHouseSize(String houseSize) {
		this.houseSize = houseSize;
	}
	public String getHousePosition() {
		return housePosition;
	}
	public void setHousePosition(String housePosition) {
		this.housePosition = housePosition;
	}
	public String getHousePrice() {
		return housePrice;
	}
	public void setHousePrice(String housePrice) {
		this.housePrice = housePrice;
	}
	public String toString(){
		return "房子大小:"+houseSize+" 房子位置:"+housePosition+" 房子价格:"+housePrice;
	}
}

如果多个bean存在相同的配置信息,Spring允许我们定义一个父,子将自动继承父的配置信息。

     <!-- 定义抽象bean -->
     <bean id="abstracthouse" class="com.mucfc.House" p:houseSize="150坪"
        p:housePosition="科苑花园" p:housePrice="15万"/>
     <!-- 继承于abstracthouse -->
      <bean id="house2" parent="abstracthouse" p:housePosition="汇景苑"/>
      <!-- 继承于abstracthouse -->
      <bean id="house3" parent="abstracthouse" p:housePrice="8万"/>

使用:  

		House house5=applicationContext.getBean("house2",House.class);
		House house6=applicationContext.getBean("house3",House.class);
		System.out.println(house5);
		System.out.println(house6);

输出结果:

房子大小:150坪 房子位置:汇景苑 房子价格:15万
房子大小:150坪 房子位置:科苑花园 房子价格:8万

    

        house1跟house2都继承自abstracthouse,Spring会将父bean的配置信息传递给子bean,如果子bean提供了父bean已有的配置信息,那么子bean的会覆盖父bean的.

       父bean的功能主要是为了简化子bean的配置,所以一般声明为abstract=“true”,表示这个不实例化为一个对应的Bean,如果用户不指定该属性为true,那么IOC容器会实例化一个名叫abstractcar的Bean。

二、依赖Bean配置

  • Spring允许用户通过depends-on属性设定Bean前置依赖的Bean ,前置依赖的Bean会在本Bean实例化之前创建好
  • 如果前置依赖于多个Bean,则可以通过逗号,空格或的方式配置Bean的名称

depends-on:

depends-on属性可以用于当前bean初始化之前显式地强制一个或多个bean被初始化。下面的例子中使用了depends-on属性来指定一个bean的依赖。

    <bean id="buyHouser" class="com.mucfc.depend.BuyHouser" /> 
    <bean id="HouseAgent" class="com.mucfc.depend.HouseAgent" depends-on="buyHouser" /> 

若需要表达对多个bean的依赖,可以在'depends-on'中将指定的多个bean名字用分隔符进行分隔,分隔符可以是逗号、空格及分号等。下面的

 <bean id="HouseAgent" class="com.mucfc.depend.HouseAgent" init-method="initHouseAgent" depends-on="buyHouser,sellHouser" /> 

例子中使用了'depends-on'来表达对多个bean的依赖。

延迟初始化bean;

在XML配置文件中,延迟初始化将通过<bean/>元素中的lazy-init属性来进行控制。例如:

    <bean id="buyHouser" class="com.mucfc.depend.BuyHouser" lazy-init="true"/> 
    <bean id="sellHouser" class="com.mucfc.depend.SellHouser" /> 

当ApplicationContext实现加载上述配置时,设置为lazy的bean-buyHouser将不会在ApplicationContext启动时提前被实例化,而not.lazy--sellHouser却会被提前实例化。**
      需要说明的是,如果一个bean被设置为延迟初始化,而另一个非延迟初始化的singleton bean依赖于它,那么当ApplicationContext提前实例化singleton bean时,它必须也确保所有上述singleton 依赖bean也被预先初始化,当然也包括设置为延迟实例化的bean。因此,如果Ioc容器在启动的时候创建了那些设置为延迟实例化的bean的实例,你也不要觉得奇怪,因为那些延迟初始化的bean可能在配置的某个地方被注入到了一个非延迟初始化singleton bean里面。**

使用范例:
1.买房人BuyHouser.java

package com.mucfc.depend;

public class BuyHouser {
private String name;

public String getName() {
	return name;
}

public void setName(String name) {
	this.name = name;
}
public void initBuyHouser(){
	System.out.println("初始化了BuyHouser");
	}
}

2、卖房人SellHouser.java

package com.mucfc.depend;

public class SellHouser {
	private String name;
	public String getName() {
		return name;
	}

	public void setName(String name) {
		this.name = name;
	}
	public void initSellHouser (){
		System.out.println("初始化了SellHouser ");
		}
}

3、房屋中介HouseAgent.java

package com.mucfc.depend;

public class HouseAgent {
private BuyHouser buyHouser;
private SellHouser sellHouser;
public BuyHouser getBuyHouser() {
	return buyHouser;
}
public void setBuyHouser(BuyHouser buyHouser) {
	this.buyHouser = buyHouser;
}
public SellHouser getSellHouser() {
	return sellHouser;
}
public void setSellHouser(SellHouser sellHouser) {
	this.sellHouser = sellHouser;
}
public void initHouseAgent(){
	System.out.println("初始化了HouseAgent");
	}
}

4、beans.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" xmlns:p="http://www.springframework.org/schema/p"
    xmlns:context="http://www.springframework.org/schema/context"
    xsi:schemaLocation="http://www.springframework.org/schema/beans  
                http://www.springframework.org/schema/beans/spring-beans-3.0.xsd  
        http://www.springframework.org/schema/context  
        http://www.springframework.org/schema/context/spring-context-3.0.xsd">

    <bean id="houseAgent" class="com.mucfc.depend.HouseAgent" init-method="initHouseAgent" depends-on="buyHouser,sellHouser" /> 
    <bean id="buyHouser" class="com.mucfc.depend.BuyHouser" init-method="initBuyHouser" lazy-init="true"/> 
    <bean id="sellHouser" class="com.mucfc.depend.SellHouser" init-method="initSellHouser" lazy-init="true"/> 

</beans>

5、测试

package com.mucfc.depend;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class Test {

	public static void main(String[] args) {
		 ApplicationContext ctx=new ClassPathXmlApplicationContext("beans.xml");
	
	}

}

结果:

注意,这里我将BuyHouser和SellHouser都设置成lazy-init="true",但是由于房屋中介HouseAgent依赖到它们两个,所以你你加载这个beans.xml文件时,还是会先把BuyHouser和SellHouser实例化。

三、引用Bean配置

代码结构:
可写成

<bean id="houseAgent1" class="com.mucfc.depend.HouseAgent" p:buyHouser="buyHouser" p:sellHouser="sellHouser"/>
   <bean id="houseAgent2" class="com.mucfc.depend.HouseAgent" p:buyHouser-ref="buyHouser" p:sellHouser-ref="sellHouser"/>

或写成

  <bean id = "houseAgent3" class = "com.mucfc.depend.HouseAgent" >
        <property name="buyHouser">
            <ref bean="buyHouser"/>
        </property>
        <property name="sellHouser">
            <ref local="sellHouser"/>
        </property>
    </bean> 

**<ref local="xx"/> **
       用"local"属性指定目标其实是指向同一文件内对应"id"属性值为此"local"值的索引"local"属性的值必须和目标bean的id属性相同。如果同一文件内没有匹配的元素,xml解析器将提示错误。同样,如果目标在同一XML文件内,使用"local"变量是最好的选择(为了尽可能早地知道错误)
**<ref bean="xx"/> **
       用"bean"属性指定目标bean是最常规的形式,这允许创建索引到任何同一个容器内的bean(无论是否在同一XML文件中)或者父级的容器内的bean。"bean"属性的值可以和目标bean的"id"属性相同,也可以和目标bean的"name"属性内的一个值相同

1 、用 local 属性指定目标 bean 可以利用 xml 解析器的能力在同一个 XML配置文件中验证 xml id 引用,没有匹配的元素,xml 解析器就会产生一个 error, 所以如果引用的 bean 在同一个 XML配置 文件中 , 那么用 local 形式是最好的选择 .
2 、可以这么说,<ref bean> 是寻找所有 XML配置文件中的 bean; <ref local> 是寻找本 xml 文件中的 bean.
3 、<ref> 提供了如下几方面的属性 :
1)bean: 在当前 Spring XML 配置文件中,或者在同一 BeanFactory(ApplicationContext) 中的其他 JavaBean 中寻找引入的BEAN.
2)local: 仅在当前 Spring XML 配置文件中寻找引入的BEAN.
如果借助于 Spring IDE, 则在编译期可以对其依赖的 JavaBean 进行验证。基于 local 方式,开发者能够使用到 XML 本身提供的优势,而进行验证。 3)parent: 用于指定其依赖的父 JavaBean 定义。

      林炳文Evankaka原创作品。转载请注明出处http://blog.csdn.net/evankaka

相关文章