继承

  Spring提供了配置信息的继承机制,可以通过为<bean>元素指定parent值重用已有的<bean>元素的配置信息。

<?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.xsd">
<bean id="baseuser" class="Demo05.BaseUser">
<property name="userid" value="1"/>
<property name="username" value="rekent"/>
<property name="password" value="123"/>
</bean> <bean id="user1" parent="baseuser">
<property name="userid" value="2"/>
</bean>
</beans>

  例如:上述我创建了一个BaseUser,其后的只要继承了BaseUser的Bean都会有BaseUser的配置信息,此时只需要重写不同的部分即可。

  注意:这里的继承是指配置信息的重用,与面向对象的继承毫无关系。另外,Spring并没有要求配置信息存在继承关系的两个Bean是统一类型的,只要具有相关属性即可。


依赖

  Spring 通过Bean之间的引用ref建立了所有Bean之间的完整依赖关系,当实例化一个Bean时,IoC容器能保证该Bean所依赖的其他Bean已经初始化完毕。

  但是有时,可能要求Bean A的初始化必须在Bean B的初始化之后,而B不是A的属性,因此无法通过向A注入B来保证首先完成B的创建。

  此时便提供了depends-on属性来指定前置依赖的Bean。例如:

<bean id="father" class=""></bean>
<bean id="son" class="" depends-on="father"></bean>

作用域

  作用域通过<bean>元素的scope属性指定,Spring支持5种作用域。

作用域 描述
singleton 一个Bean定义对应唯一一个对象实例,Bean以单实例的方式存在(默认)
prototype 一个Bean定义对应多个对象实例,每次调用getBean()时,就创建一个新实例(开销大,不推荐)
request (仅在基于Web的SpringApplicationContext情形下有效)在一次Http请求中,一个Bean定义对应一个实例,即每个请求都会有各自的Bean。
session (仅在基于Web的SpringApplicationContext情形下有效)在一个HtppSession中,一个Bean定义对应一个实例。
global session (仅在基于Web的SpringApplicationContext情形下有效)在一个全局的Http Session中,一个Bean定义对应一个实例。

自动装配(源于他人,原文链接:http://www.cnblogs.com/sysman/p/4485199.html

可以使用bean元素的autowire属性指定自动装配的类型,spring支持如下类型:

自动装配的类型 描述
no/default autowire="no"指定spring不使用自动装配,需要手动装配
byName 按照bean属性的名字从spring容器中找同名的bean进行注入,适用于setter注入
byType 按照bean属性的类型从spring容器中找相同类型的bean进行注入,适用于setter注入
constructor 按照类型装配,跟byType类似.适用于构造器参数注入

下面我们将分别讲解着四种装配类型

不使用自动装配-no

我们之前讲解的所有的例子都属于这种类型.在这种情况下所有bean的装配都是手动进行的.我们再用一个例子复习下

1.新建包com.tutorialspoint.autowire,并在包中新建Cat.java、Dog.java、Duck.java.后面所有例子都会用到这三个类:

//Cat.java
package com.tutorialspoint.autowire; public class Cat { public void sayHi(){
System.out.println("miao miao ... ");
}
} //Dog.java
package com.tutorialspoint.autowire; public class Dog { public void sayHi(){
System.out.println("wang wang ... ");
}
} //Duck.java
package com.tutorialspoint.autowire; public class Duck { public void sayHi(){
System.out.println("ga ga ... ");
}
}

2.新建包com.tutorialspoint.autowire.no,并在包中新建Zoo.java,内容如下:

package com.tutorialspoint.autowire.no;

import com.tutorialspoint.autowire.*;

public class Zoo {

    private Cat cat;
private Dog dog;
private Duck duck; public void setCat(Cat cat) {
this.cat = cat;
}
public void setDog(Dog dog) {
this.dog = dog;
}
public void setDuck(Duck duck) {
this.duck = duck;
} public void print(){ if(cat==null){
System.out.println("cat is null");
}else{
cat.sayHi();
} if(dog==null){
System.out.println("dog is null");
}else{
dog.sayHi();
} if(duck==null){
System.out.println("duck is null");
}else{
duck.sayHi();
}
} }

3.在src目录下新建autowire_no.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-3.0.xsd"> <bean id="cat" class="com.tutorialspoint.autowire.Cat"></bean>
<bean id="dog" class="com.tutorialspoint.autowire.Dog"></bean>
<bean id="duck" class="com.tutorialspoint.autowire.Duck"></bean> <bean id="zoo" class="com.tutorialspoint.autowire.no.Zoo">
<property name="cat" ref="cat"></property>
<property name="dog" ref="dog"></property>
</bean> </beans>

4.在com.tutorialspoint.autowire.no包中新建MainApp.java.内容如下:

package com.tutorialspoint.autowire.no;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class MainApp { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("autowire_no.xml"); Zoo zoo = (Zoo) context.getBean("zoo"); zoo.print();
}
}

5.运行程序,检查结果:

通过上面的程序我们可以得出如下结论:

手动装配bean,bean的所有依赖项都要在bean元素中明确指定.如果不进行指定spring容器就不会注入该属性.

按照名字进行自动装配-byName

在byName装配方式下,spring首先会反射autowire="byName"的bean,得到bean中的所有属性名(根据setter推算),然后从容

器中寻找同名的bean,最后把找到的bean注入到当前bean中.我们还是用代码说话:

1.新建包com.tutorialspoint.autowire.byname,并在包中新建Zoo.java类.内容如下:

package com.tutorialspoint.autowire.byname;

import com.tutorialspoint.autowire.Cat;
import com.tutorialspoint.autowire.Dog;
import com.tutorialspoint.autowire.Duck; public class Zoo { private Cat cat;
private Dog dog;
private Duck duck;
// 自动装配并不适用于原始类型.这时候我们可以对该属性进行手动装配
private String zooName; public void setZooName(String zooName) {
this.zooName = zooName;
} public void setCat(Cat cat) {
this.cat = cat;
} public void setDog(Dog dog) {
this.dog = dog;
} public void setDuck(Duck duck) {
this.duck = duck;
} public void print() { if (cat == null) {
System.out.println("cat is null");
} else {
cat.sayHi();
} if (dog == null) {
System.out.println("dog is null");
} else {
dog.sayHi();
} if (duck == null) {
System.out.println("duck is null");
} else {
duck.sayHi();
} System.out.println(zooName);
} }

2.在src目录下新建autowire_byName.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-3.0.xsd"> <!-- 名为zoo的bean在按照byName进行装配的时候,可以适配名字为cat、dog
的bean,不能适配名字为duck1的bean。所以最终zoo中会注入进cat和dog
不会注入duck. -->
<bean id="cat" class="com.tutorialspoint.autowire.Cat"></bean>
<bean id="dog" class="com.tutorialspoint.autowire.Dog"></bean>
<bean id="duck1" class="com.tutorialspoint.autowire.Duck"></bean> <bean id="zoo" class="com.tutorialspoint.autowire.byname.Zoo" autowire="byName">
<!-- 由于自动装配仅适用于引用类型,普通类型还需要手动进行注入 -->
<property name="zooName" value="international_zoo"></property>
</bean> </beans>

3.在包com.tutorialspoint.autowire.byname中新建MainApp.java.内容如下:

package com.tutorialspoint.autowire.byname;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class MainApp { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("autowire_byName.xml"); Zoo zoo = (Zoo)context.getBean("zoo"); zoo.print();
}
}

4.运行代码,检查结果:

分析结果可以看到cat和dog已经按照属性名字自动装配到了zoo中.duck1由于没有匹配的属性名所以没有进行装配.zooName是

我们手动进行注入的。

按照类型进行自动装配-byType

在byType装配方式下,spring首先会反射autowire="byType"的bean,得到bean属性的返回类型,然后去spring容器中按照类

型去匹配,最后把匹配到的bean注入到当前bean中.看个例子就明白了:

1.新建包com.tutorialspoint.autowire.bytype,并在包中新建Zoo.java类,内容如下:

package com.tutorialspoint.autowire.bytype;

import com.tutorialspoint.autowire.Cat;
import com.tutorialspoint.autowire.Dog;
import com.tutorialspoint.autowire.Duck; public class Zoo { private Cat cat;
private Dog dog;
private Duck duck;
// 自动装配并不适用于原始类型.这时候我们可以对该属性进行手动装配
private String zooName; public void setZooName(String zooName) {
this.zooName = zooName;
} public void setCat(Cat cat) {
this.cat = cat;
} public void setDog(Dog dog) {
this.dog = dog;
} public void setDuck(Duck duck) {
this.duck = duck;
} public void print() { if (cat == null) {
System.out.println("cat is null");
} else {
cat.sayHi();
} if (dog == null) {
System.out.println("dog is null");
} else {
dog.sayHi();
} if (duck == null) {
System.out.println("duck is null");
} else {
duck.sayHi();
} System.out.println(zooName);
} }

2.在src目录下新建配置文件autowire_byType.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-3.0.xsd"> <!-- 名为zoo的bean在按照byType进行装配的时候,可以适配名字为cat1、dog1、duck1
的bean的类型。所以最终cat1、dog1、duck1会被分别注入进zoo的cat、dog、duck属性。
在按照类型进行装配的时候,如过有两个bean的类型符合的话,spring就不知道最终该使用哪个,这时候我们
可以使用primary="true"告诉spring优先使用本bean
-->
<bean id="cat1" class="com.tutorialspoint.autowire.Cat" primary="true"></bean>
<bean id="cat2" class="com.tutorialspoint.autowire.Cat"></bean>
<bean id="dog1" class="com.tutorialspoint.autowire.Dog"></bean>
<bean id="duck1" class="com.tutorialspoint.autowire.Duck"></bean> <bean id="zoo" class="com.tutorialspoint.autowire.bytype.Zoo" autowire="byType">
<!-- 由于自动装配仅适用于引用类型,普通类型还需要手动进行注入 -->
<property name="zooName" value="international_zoo"></property>
</bean> </beans>

3.在包com.tutorialspoint.autowire.bytype中新建MainApp.java。内容如下:

package com.tutorialspoint.autowire.bytype;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class MainApp { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("autowire_byType.xml"); Zoo zoo = (Zoo)context.getBean("zoo"); zoo.print();
}
}

4.运行程序,检查结果:

构造器参数类型自动装配-constructor

在constructor自动装配模式下,spring首先会反射bean的构造函数,得出构造函数的参数的类型,然后起spring容器中匹配合适的

类型的bean,最后使用构造器参数注入的方法把符合的bean注入到当前bean中。看代码:

1.新建包com.tutorialspoint.autowire.constructor,并在包中新建Zoo.java。内容如下:

package com.tutorialspoint.autowire.constructor;

import com.tutorialspoint.autowire.*;

public class Zoo {

    private Cat cat;
private Dog dog;
private Duck duck;
private String zooName; public Zoo(Cat cat, Dog dog, Duck duck, String zooName) {
this.cat = cat;
this.dog = dog;
this.duck = duck;
this.zooName = zooName;
} public void print() { if (cat == null) {
System.out.println("cat is null");
} else {
cat.sayHi();
} if (dog == null) {
System.out.println("dog is null");
} else {
dog.sayHi();
} if (duck == null) {
System.out.println("duck is null");
} else {
duck.sayHi();
} System.out.println(zooName);
} }

2.在src目录下新建autowire_constructor.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-3.0.xsd"> <!-- constructor跟byType十分相似.
名为zoo的bean在按照constructor进行装配的时候,可以适配名字为cat1、dog1、duck1
的bean的类型。所以最终cat1、dog1、duck1会被分别注入进zoo的cat、dog、duck属性(使用
构造器参数进行注入)。
在按照constructor进行装配的时候,如过有两个bean的类型符合的话,spring就不知道最终该使用哪个,这时候我们
可以使用primary="true"告诉spring优先使用本bean
-->
<bean id="cat1" class="com.tutorialspoint.autowire.Cat" primary="true"></bean>
<bean id="cat2" class="com.tutorialspoint.autowire.Cat"></bean>
<bean id="dog" class="com.tutorialspoint.autowire.Dog"></bean>
<bean id="duck1" class="com.tutorialspoint.autowire.Duck"></bean> <bean id="zoo" class="com.tutorialspoint.autowire.constructor.Zoo" autowire="constructor">
<!-- 由于自动装配仅适用于引用类型,普通类型还需要手动进行注入 -->
<constructor-arg name="zooName" value="international_zoo"></constructor-arg>
</bean> </beans>

3.在com.tutorialspoint.autowire.constructor包中新建MainApp.java。内容如下:

package com.tutorialspoint.autowire.constructor;

import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext; public class MainApp { public static void main(String[] args) { ApplicationContext context = new ClassPathXmlApplicationContext("autowire_constructor.xml"); Zoo zoo = (Zoo)context.getBean("zoo"); zoo.print();
}
}

4.运行程序,检查结果:

如果使用sping的自动装配,本人不推荐使用xml的配置方式.最好使用注解的配置方式。原因如下:

1.基于xml的自动装配粒度态度。默认会装配所有符合条件的bean.不能指定哪个属性不进行自动装配

2.不能指定哪些属性必须进行装配,否则抛出异常.

以上两点使用spring的注解配置元数据都是可以做到的。我们下节就讲解spring注解配置元数据。

Spring 学习笔记(五)—— Bean之间的关系、作用域、自动装配的更多相关文章

  1. Spring学习记录(四)---bean之间的关系:继承、依赖

         继承 这里说的继承和java的继承是不一样的,不是父类子类.但思想很相似,是父bean和子bean 1.父bean是一个实例时.它本身是一个完整的bean 2.父bean是模板,抽象bean ...

  2. 3.spring:自动装配/Bean之间的关系/作用域/外部文件/spel/

    1.自动装配/手动装配 xml配置文件里的bean自动装配 Spring IOC 容器里可以自动的装配Bean,需要做的仅仅是在<bean>的autowire属性里面指定自动装配模式 -& ...

  3. Spring学习笔记(2)——Bean的配置

    要使应用程序中的Spring容器成功启动,需要以下三个方面的条件都具备: 1.Spring框架的类包都已经放到应用程序的类路径下 2.应用程序为Spring提供完备的Bean配置信息 3.Bean的类 ...

  4. Spring学习笔记(3)——Bean的注入方式

    依赖注入 依赖注入支持属性注入.构造函数注入.工厂注入. 属性注入: 属性注入即通过setXxx()方法注入Bean的属性值或依赖对象 属性注入要求Bean提供一个默认的构造函数(无参构造函数),并为 ...

  5. Spring学习笔记--注入Bean属性

    这里通过一个MoonlightPoet类来演示了注入Bean属性property的效果. package com.moonlit.myspring; import java.util.List; im ...

  6. Spring4.0学习笔记(3) —— Spring_Bean之间的关系

    1.继承关系 bean-relation.xml <?xml version="1.0" encoding="UTF-8"?> <beans ...

  7. Spring学习笔记——02 Bean的命名及实例化

    一.Bean的命名 前一篇讲到IoC是一个管理Bean的容器,Bean多数情况下都是通过XML文件进行配置的,其中Bean的命名有以下几种方式,现在梳理一下. 1. 不指定id,只配置类名 <b ...

  8. Spring学习记录(五)---bean的作用域scope

    作用域:singleton:单例,整个应用中只创建一个实例(默认) prototype:原型,每次注入时都新建一个实例 session:会话,每个会话创建一个实例 request:请求,每个请求创建一 ...

  9. Spring学习笔记之bean配置

    1.命名bean 每个bean都有一个或者多个的的标识符.这些标识符必须在加载他们的容器里边唯一.一个bean经常有且只有一个标识符,但是如果需要超过一个的名字,可以考虑额外的别名. 基于xml的配置 ...

  10. Spring学习笔记之Bean的实例化

    一.bean的实例化方法有3种, 1.构造器实例化 2.静态工厂方法实例化 3.实例工厂方法实例化 二.用构造器来实例化 <bean id="ShunDao" class=& ...

随机推荐

  1. Deep Learning Libraries by Language

    Deep Learning Libraries by Language Tweet         Python Theano is a python library for defining and ...

  2. 一篇RxJava友好的文章(一)

    转载请标明出处: http://blog.csdn.net/forezp/article/details/52886700 本文出自方志朋的博客 Rxjava在目前的开发中已经是如火如荼,非常的流行, ...

  3. v-for的简单实现

    <!DOCTYPE html> <html> <head> <meta charset="UTF-8"> <title> ...

  4. CSU 1216异或最大值 (0-1 trie树)

    Description 给定一些数,求这些数中两个数的异或值最大的那个值 Input 多组数据.第一行为数字个数n,1 <= n <= 10 ^ 5.接下来n行每行一个32位有符号非负整数 ...

  5. CentOS 系统 Docker 的命令大全

    本文记录 CentOS 系统 Docker 的命令大全 命令 docker ps 显示运行中的容器 docker ps -a 显示所有容器 docker start 容器名称 启动容器 docker ...

  6. mysql架构和历史

    存储引擎 查看: show table status like 'bigcourse'; 结果: +-----------+--------+---------+------------+------ ...

  7. datatable常用设置

    bSort: false, // 是否排序功能 bFilter: false, // 过滤功能 bPaginate: true, // 翻页功能 bInfo: true, // 页脚信息 bProce ...

  8. thinkphp5,单图,多图,上传

    /** * 上传单图 */ function upload($path, $filename) { $file = request()->file($filename); $info = $fi ...

  9. linux 特殊命令(一)

    1.ifconfig 网卡配置:ifconfig  [网络设备] [参数] 1) up 启动指定网络设备/网卡. 2) down 关闭指定网络设备/网卡.该参数可以有效地阻止通过指定接口的IP信息流, ...

  10. PHP 微信公众号真正正确的客服头像上传

    首先我们来看官方文档 这TM的搞笑呢 什么破玩意儿! 需要条件 1 需要有一个客服的账号 (废话) 2 一致jpg格式的图片(扯蛋) 完整流程 1 获取access_token 2获取账号 3 $ur ...