自动装配
- Spring IOC 容器可以自动装配 Bean. 需要做的仅仅是在 的 autowire 属性里指定自动装配的模式
- byType(根据类型自动装配): 若 IOC 容器中有多个与目标 Bean 类型一致的 Bean. 在这种情况下, Spring
将无法判定哪个 Bean 最合适该属性, 所以不能执行自动装配. - byName(根据名称自动装配): 必须将目标 Bean 的名称和属性名设置的完全相同.
- constructor(通过构造器自动装配): 当 Bean 中存在多个构造器时, 此种自动装配方式将会很复杂. 不推荐使用
bean-autowire.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"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="address" class="com.anqi.autowire.Address"
p:city="Taiyuan" p:street="HonggouStreet"></bean>
<bean id="car" class="com.anqi.autowire.Car" p:brand="ChangAn"
p:price="250000"></bean>
<!--
可以使用 autowire 属性指定自动装配的方式
byName 根据 bean 的名字和当前 bean 的 setter 风格的属性名进行自动装配, 若有匹配的则进行自动装配
byType 根据 bean 的类型和当前 bean 的类型进行装配 若 IOC 容器里有1个以上的类型匹配的bean,则抛异常
-->
<bean id="person" class="com.anqi.autowire.Person"
p:name="AnQi" p:address-ref="address" autowire="byName"> </bean>
<bean id="person2" class="com.anqi.autowire.Person"
p:name="AnQi2" p:address-ref="address" autowire="byType"> </bean>
</beans>
Person.java
package com.anqi.autowire;
public class Person {
private String name;
private Address address;
private Car car;
//...setter 、getter、toString
}
Address.java
package com.anqi.autowire;
public class Address {
private String city;
private String street;
//...setteri 、getter、toString
}
Car.java
package com.anqi.autowire;
public class Car {
private String brand;
private double price;
//...setteri 、getter、toString
}
Main.java
package com.anqi.autowire;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
public class Main {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("bean-autowire.xml");
Person p = (Person) ctx.getBean("person");
System.out.println(p);
//Person [name=AnQi, address=Address [city=Taiyuan, street=HonggouStreet],
//car=Car [brand=ChangAn, price=250000.0]]
p = (Person) ctx.getBean("person2");
System.out.println(p);
//Person [name=AnQi2, address=Address [city=Taiyuan, street=HonggouStreet],
//car=Car [brand=ChangAn, price=250000.0]]
}
}
bean 之间的关系
继承关系
- Spring 允许继承 bean 的配置, 被继承的 bean 称为父 bean. 继承这个父 Bean 的 Bean 称为子 Bean
- 子 Bean 从父 Bean 中继承配置, 包括 Bean 的属性配置
- 子 Bean 也可以覆盖从父 Bean 继承过来的配置
- 父 Bean 可以作为配置模板, 也可以作为 Bean 实例. 若只想把父 Bean 作为模板, 可以设置 的abstract 属性为 true, 这样 Spring 将不会实例化这个 Bean
- 并不是 元素里的所有属性都会被继承. 比如: autowire, abstract 等.
- 也可以忽略父 Bean 的 class 属性, 让子 Bean 指定自己的类, 而共享相同的属性配置. 但此时 abstract 必须设为 true
依赖关系
- Spring 允许用户通过 depends-on 属性设定 Bean 前置依赖的Bean,前置依赖的 Bean 会在本 Bean 实例化之前创建好
- 如果前置依赖于多个 Bean,则可以通过逗号,空格或的方式配置 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"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd">
<!-- 抽象 bean : bean 的 abstract 属性为 true 的bean, 这样的 bean 不能被 IOC 容器实例化
只能被用来继承配置 -->
<!--<bean id="address" class="com.anqi.autowire.Address" p:city="Taiyuan"
p:street="Wulongkou"></bean> -->
<!-- 若某父类 bean 的 class 属性没有被指定, 则该 bean 必须是一个抽象 bean -->
<bean id="address" p:city="Taiyuan"
p:street="Wulongkou" abstract="true"></bean>
<!-- bean 配置的继承 : 使用 bean 的 parent 属性指定继承哪个 bean 的配置 -->
<bean id="address2" class="com.anqi.autowire.Address" p:city="Xinzhou"
parent="address"></bean>
<bean id="car" class="com.anqi.autowire.Car" p:brand="Benci"
p:price="290000"></bean>
<!-- 要求在配置 Person 时, 必须有一个关联的 car,换句话说 Person 这个 bean 依赖于 Car 的 bean -->
<bean id="person" class="com.anqi.autowire.Person" p:name="安琪"
p:address-ref="address2" depends-on="car" p:car-ref="car"></bean>
</beans>
Bean 的作用域
- 在 Spring 中, 可以在 元素的 scope 属性里设置 Bean 的作用域.
- 默认情况下, Spring 只为每个在 IOC 容器里声明的 Bean 创建唯一一个实例, 整个 IOC 容器范围内都能共享该实例:所有后续的 getBean() 调用和 Bean 引用都将返回这个唯一的 Bean 实例.该作用域被称为 singleton, 它是所有 Bean 的默认作用域
<!--
使用 bean 的 scope 属性来配置 bean 的作用域
singleton : 默认值, 容器的初始时创建 bean 对象, 在整个容器的生命周期只能创建
这一个 bean, 单例的。无参构造器在容器启动时执行。
prototype : 原型的, 容器初始化时不创建 bean 的实例, 而在每次请求时都创建一个
新的 bean 实例, 并返回
-->
<bean id="car" class="com.anqi.autowire.Car">
<property name="brand" value="fute"></property>
<property name="price" value="2500000"></property>
</bean>
<bean id="car2" class="com.anqi.autowire.Car" scope="prototype">
<property name="brand" value="fute2"></property>
<property name="price" value="25000002"></property>
</bean>
public static void main(String[] args) {
ApplicationContext ctx =
new ClassPathXmlApplicationContext("bean-scope.xml");
Car car = (Car) ctx.getBean("car");
Car car_ = (Car) ctx.getBean("car");
System.out.println(car == car_);
//Constructor.... 不创建对象也会在 ioc 容器启动时自己执行
//true
Car car2 = (Car) ctx.getBean("car2");
Car car3 = (Car) ctx.getBean("car2");
System.out.println(car2 == car3);
//Constructor.... 创建对象时调用
//false
}
使用外部属性文件
- 在配置文件里配置 Bean 时, 有时需要在 Bean 的配置里混入系统部署的细节信息(例如: 文件路径, 数据源配置信息等).而这些部署细节实际上需要和 Bean 配置相分离
- Spring 提供了一个 PropertyPlaceholderConfigurer 的 BeanFactory 后置处理器, 这个处理器允许用户将 Bean 配置的部分内容外移到属性文件中. 可以在 Bean 配置文件里使用形式为 ${var} 的变量, PropertyPlaceholderConfigurer 从属性文件里加载属性, 并使用这些属性来替换变量.
Spring 还允许在属性文件中使用 ${propName},以实现属性之间的相互引用
例子–>Spring 通过外部属性导入来配置 c3p0
准备工作
导入
c3p0-0.9.5.2.jar
mchange-commons-java-0.2.11.jar
mysqljdbc.jar
使用context命名空间
外部属性文件
user=root
password=1995
driverclass=com.mysql.jdbc.Driver
jdbcurl=jdbc:mysql:///db_javaee
<?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:context="http://www.springframework.org/schema/context"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">
<!-- 使用 context 命名空间, 导入属性文件 -->
<context:property-placeholder location="classpath:db.properties"/>
<!--
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="user" value="root"></property>
<property name="password" value="1995"></property>
<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
<property name="jdbcUrl" value="jdbc:mysql:///db_javaee"></property> -->
<!--
准备工作-导入:
c3p0-0.9.5.2.jar
mchange-commons-java-0.2.11.jar
mysqljdbc.jar
这样的优势是:项目很大的时候,bean 很多,若要修改数据库等配置信息,或者修改数据库
寻找到这个 bean 花费大量时间, 若采取导入属性文件的方式, 我们直接
修改 db.properties 更为方便
除了这些属性, 还有最大连接数等属性可供设置
-->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource" >
<!-- 使用外部化属性文件的属性 -->
<property name="user" value="${user}"></property>
<property name="password" value="${password}"></property>
<property name="driverClass" value="${driverclass}"></property>
<property name="jdbcUrl" value="${jdbcurl}"></property>
</bean>
</beans>
public class Main {
public static void main(String[] args) throws SQLException {
ApplicationContext ctx = new ClassPathXmlApplicationContext("bean-properties.xml");
DataSource dataSource = (DataSource) ctx.getBean("dataSource");
System.out.println(dataSource.getConnection());
}
}