1)继承:
如果多个bean存在相同的配置信息,Spring允许我们定义一个父,子将自动继承父的配置信息。
如下:
<bean
class
=
"com.baobaobao.Car"
id=
"car1"
p:color=
"红色"
p:price=
"200000"
p:brand=
"奥迪"
>
<bean
class
=
"com.baobaobao.Car"
id=
"car2"
p:color=
"白色"
p:price=
"200000"
p:brand=
"奥迪"
></bean></bean>
这两个bean的配置存在大量重复信息,可以通过父子bean很好的消除这种重复配置:
<bean
class
=
"com.baobaobao.Car"
id=
"abstractcar"
p:color=
"黑色"
p:price=
"200000"
p:brand=
"奥迪"
abstract
=
"true"
>
<bean id=
"car1"
p:color=
"红色"
parent=
"abstractcar"
>
<bean id=
"car2"
p:color=
"白色"
parent=
"abstractcar"
>
</bean>
</bean></bean>
Car1跟Car2都继承自abstractcar,Spring会将父bean的配置信息传递给子bean,如果子bean提供了父bean已有的配置信息,那么子bean的会覆盖父bean的
父bean的功能主要是为了简化子bean的配置,所以一般声明为abstract=“true”,表示这个不实例化为一个对应的Bean,如果用户不指定该属性为true,那么IOC容器会实例化一个名叫abstractcar的Bean。
2)依赖
一般情况使用即可建立一个bean对其他bean的依赖关系,Spring负责管理这些Bean的关系,当实例化一个Bean时,Spring保证该Bean所依赖的其他Bean已经初始化。
书上举了一个比较经典而特殊的例子:
例如一个系统,有一个SystemSetting类,
| public class SystemSetting { public static int SESSION_TIMEOUT= 30 ; public static int REFRESH_CYCLE= 60 ; ..... } |
里面有一些系统参数的默认值。
而这个系统有个管理后台,可以修改这些值保存到中,而有一个SysInit类,在初始化的时候从数据中拿到值,修改SystemSetting的参数
| public class SysInit { public SysInit(){ SystemSetting.SESSION_TIMEOUT= 100 ; SystemSetting.REFRESH_CYCLE= 120 ; } } |
| public class CacheManager { public CacheManager(){ Timer timer = new Timer(); TimerTask cacheTask = new CacheTask(); timer.schedule(cacheTask, 0 ,SystemSetting.REFRESH_CYCLE); } } |
使用depends-on属性可以指定Bean的前置依赖Bean,前置依赖Bean会在本Bean实例化之前创建好:
| <bean class = "com.baobaobao.CacheManager" id= "manager" depends-on= "sysinit" > <bean class = "com.baobaobao.SysInit" id= "sysinit" > </bean></bean> |
3)引用
假如一个bean要引用另一个bean的配置的id属性值,可以使用如下配置:
| <bean class = "com.baobaobao.Car" id= "car" > <bean class = "com.baobaobao.Boss" id= "boss" > <property name= "carId" > <idref bean= "car" > </idref> </property></bean></bean> |
如果引用者与被引用者位于同一个配置文件中,可以使用,这是IDE的XML分析器就可以在开发期发现引用错误了。
Java实例的属性值可以有很多种数据类型、基本类型值、字符串类型、java实例甚至其他的Bean实例、java集合、数组等。所以Spring允许通过如下几个元素为Bean实例的属性指定值:
value
ref
bean
list、set、map、props
一、value:设置普通属性值
<value.../>元素用于指定字符串类型、基本类型的属性值。Spring使用XML解析器来解析出这些数据,然后利用java.beans.PropertyEdior完成类型转换:从java.lang.String类型转换为所需的参数值类型。如果目标类型是基本数据类型,通常都是可以正确转换。
在Spring配置文件中使用<value.../>元素来为这两个属性指定属性值。
12 3 4
通过上面可以知道<value.../>元素主要用于传入字符串、基本类型的属性值。
二、ref:配置合作者
<value.../>主要是配置基本类型的属性值,但是如果我们需要为Bean设置属性值是另一个Bean实例时,这个时候需要使用<ref.../>元素。使用<ref.../>元素可以指定如下两个属性。
bean:引用不在同一份XML配置文件中的其他Bean实例的id属性值。
local:引用同一份XML配置文件中的其他Bean实例的id属性值。
12 3 4
通过property增加ref属性,一样可以将另一个Bean的引用设置成axe属性值。这样写的效果和使用<ref.../>属性一样,而且不需要区分是使用bean属性还是local属性,所以推荐这种写法。
2.1、使用自动装配注入合作者bean
Spring支持自动装配Bean与Bean之间的依赖关系,也就是说我们无需显示的指定依赖Bean。由BeanFactory检查XML配置文件内容,根据某种规则,为主调Bean注入依赖关系。
Spring的自动装配机制可以通过<bean.../>元素的default-autowire属性指定,也可以通过<bean.../>元素的autowire属性指定。
自动装配可以减少配置文件的工作量,但是它降低了依赖关系的透明性和清晰性,所以一般来说在较大部署环境中不推荐使用,显示配置合作者能够得到更加清晰的依赖关系。Spring提供了如下几种规则来实现自动装配。
no:不适用自动装配。Bean依赖必须通过ref元素定义。
byName:根据属性名自动装配。BeanFactory查找容器中的全部Bean,找出其中id属性与属性同名的Bean来完成注入。如果没有找到匹配的Bean实例,则Spring不会进行任何注入。
byType:根据属性类型自动装配。BeanFactory查找容器中的全部Bean,如果正好有一个与依赖属性类型相同的Bean,就自动注入这个属性;但是如果有多个这样的Bean,就会抛出一个异常。如果没有匹配的Bean,则什么都不会发生,属性就不会被设置。如果需要无法自动装配时抛出异常,则设置dependency-check=”objects”。
constructor:与不Type类似,区别是用于构造注入的参数。
Autodetect:BeanFactory根据Bean内部结构,决定使用constructor或者byType。如果找到一个默认的构造函数,则使用byTe。
byName规则
byTyep规则是指通过名字注入依赖关系,假如Bean A的实现类里面包含setB()方法,而Spring的配置文件恰好包含一个id为b的Bean,则Spring容器就会将b实例注入Bean A中。如果容器中没有名字匹配的Bean,Spring则不会做任何事情。
12 3 4
上面的配置文件指定了byName规则。则com.app.service.impl.Chinese类中提供如下的依赖注入方法:
1 /* 2 * 依赖关系必须的setter方法,因为需要通过名字自动装配 3 * 所以setter方法必须提供set+Bean名,Bean名的首字母大写 4 * @param dog 设置的dog值 5 */ 6 public void setGundog(Dog dog){ 7 this.dog = dog; 8 }
byType规则
byType规则是根据类型匹配注入依赖关系。假如A实例有setB(B b)方法,而Spring配置文件中恰好有一个类型B的Bean实例,容器为A注入类型匹配的Bean实例。如果容器中存在多个B的实例,则会抛出异常,如果没有B实例,则不会发生任何事情。
12 3 4
针对上面的配置文件Chinese类有如下方法。
1 /** 2 * 依赖关系必须的setter方法 3 * 因为使用按类型自动装配,setter方法的参数类型与容器的Bean的类型相同 4 * 程序中的Gundog实现了Dog接口 5 * @param dog传入的dog对象 6 */ 7 public void setDog(Dog dog){ 8 this.dog = dog; 9 }
当一个Bean即使用自动装配依赖,又使用ref显示依赖时,则显示指定的依赖就会覆盖自动装配。
在默认的情况下,Spring会自动搜索容器中的全部Bean,并对这些Bean进行判断,判断他们是否满足自动装配的条件,如果满足就会将该Bean注入目标Bean实例中。如果我们不想让Spring搜索容器中的全部Bean,也就是说,我们需要Spring来判断哪些Bean需要搜索,哪些Bean不需要搜索,这个时候就需要用到autowire-candidate属性。通过为<bean.../>元素设置autowire-candidate=”false”,即可将该Bean限制在自动装配范围之外,容器在查找自动装配对象时将不考虑该Bean。
三、Bean:注入嵌套Bean
如果某个Bean所依赖的Bean不想被Spring容器直接访问,则可以使用嵌套Bean。<bean.../>元素用来定义嵌套Bean,嵌套Bean只对嵌套它的外部Bean有效,Spring容器无法直接访问嵌套Bean,因此在定义嵌套Bean时是无需指定id属性的。
12 3 7 98
四、list、set、map、props
<value.../>元素是注入基本数据类型和String类型的,但是如果某个Bean的属性是集合呢?这个时候我们就需要使用集合元素,<list.../>、<set.../>、<map.../>和<props.../>元素分别用来设置类型list、set、map和Properties的集合属性值。
先看下面java类:
1 public class Chinese implements Person{ 2 3 //下面是一系列的集合属性 4 private Listschools; 5 private Map scores; 6 private Map phaseAxes; 7 private Properties health; 8 private Set axe; 9 private String[] books; 10 11 public List getSchools() { 12 return schools; 13 } 14 15 public void setSchools(List schools) { 16 this.schools = schools; 17 } 18 19 public Map getScores() { 20 return scores; 21 } 22 23 public void setScores(Map scores) { 24 this.scores = scores; 25 } 26 27 public Map getPhaseAxes() { 28 return phaseAxes; 29 } 30 31 public void setPhaseAxes(Map phaseAxes) { 32 this.phaseAxes = phaseAxes; 33 } 34 35 public Properties getHealth() { 36 return health; 37 } 38 39 public void setHealth(Properties health) { 40 this.health = health; 41 } 42 43 public Set getAxe() { 44 return axe; 45 } 46 47 public void setAxe(Set axe) { 48 this.axe = axe; 49 } 50 51 public String[] getBooks() { 52 return books; 53 } 54 55 public void setBooks(String[] books) { 56 this.books = books; 57 } 58 59 public void useAxe() { 60 61 } 62 }
1 26 7 8 9 10 11 12 5813 19 2014
18小学 15中学 16大学 1721 26 27 2829 33 34 3536 41 4237 40正常 38175 3943 49 5044 48普通字符串 4546 47 51 5752
56java 编程思想 53思考致富 54将才 55
从上面的配置文件中可以看出,Spring对list属性和数组属性的处理是一样的。
当我们使用<list.../>、<set.../>、<map.../>等元素配置集合属性时,我们还需要手动配置集合元素。由于集合元素又可以是基本类型值、引用容器中的其他Bean、嵌套Bean和集合属性等。所以这些元素又可以接受如下子元素:
value:指定集合元素是基本数据类型或者字符类型值。
ref:指定集合元素师容器中另一个Bean实例。
bean:指定集合元素是一个嵌套Bean。
list、set、map、props:指定集合元素值又是集合。