Bean的定义:
1. 从本质上讲,Spring是一个大的工厂,而Bean则是工厂中的产品。
2. 对于XML中的<beans>基本标签,主要是设置的是bean的默认属性。
<bean>中的作用域,用scope设置,有五种类型
1. singleton:单利模式,只有一个对象
2. prototype:原型模式,每次同个getBean()获取都是一个新的对象
3. request::只有web中可以使用,每次使用request请求 就会产生一个新的实例,request完成后,实例也会消失。
4. session:只有web对象可以使用,使用session就会产生一个新的实例
5. global session:?
<bean>设置对象依赖,有两种方法:
//(1) <bean id="" class="" scope=""> <property name="" value=""></property> <property name="" ref=""></property> </bean> //(2) <bean id="" class="" scope=""> <contructor index="" value=""></property> <contructor index="" ref=""></property> </bean>如果对象是同一个XML,使用ref,如果不是,使用local
<bean id="" class="" scope=""> <property name=""> <ref local=""/> </property> <property name="" ref=""></property> </bean> 注入集合 <bean id="listtext" class="ok.ListTest" scope="singleton"> <property name="list"> <list> <value>小薛</value> <value>高中</value> <value>初中</value> </list> </property> <property name="scores"> <map> <entry key="数学1" value="89" /> <entry key="数学2" value="89" /> <entry key="数学3" value="89" /> <entry key="数学4" value="89" /> </map> </property> <property name="menscores"> <map> <entry key="1" value-ref="men" /> <entry key-ref="" value-ref=""> </map> </property> <property name="set"> <set> <value>你好</value> <bean class="ok.Person"/> <ref local="men"/> </set> </property> <property name="books"> <list> <value>小薛</value> <value>高中</value> <value>初中</value> <value>小薛</value> <value>高中</value> <value>初中</value> </list> </property> </bean>
组合属性设置
<bean id="a" class=""> <property name="foo.bar.x.y" value="xxx"></property> </bean>相当于设置 a.getFoo().getBar().getX().setY(xxx)Spring中Bean和JavaBean的区别:
1. 用处不同,传统的JavaBean更多的作为值对象传递参数;Spring中的Bean无处不在,任何应用组建都称为bean
2. 写法不同:JavaBean作为值传递,一般都会设置set和get方法;而Bean一般只需要为设置值的注入设置set方法。
3 生命周期不同,传统的JavaBean作为值传递,没有生命周期,而Bean有Spring管理,有生命周期。
通常的情况下, 组件和组件之间的耦合,采用依赖注入管理,但是普通的JavaBean属性值,应该直接在代码中设置。
Bean的创建有三种
1. 构造器创建
2. 调用静态工厂方法创建
3. 调用非静态工厂方法创建
静态工厂方法创建和非静态
<!-- 静态工厂 --> <bean id="wowan" class="bean.FactoryClass" factory-method="getPeople"> <constructor-arg value="0"/> </bean> <bean id="man" class="bean.FactoryClass" factory-method="getPeople"> <constructor-arg value="2"/> </bean> <!-- 非静态工厂 --> <bean id="factory" class="bean.FactoryClass"></bean> <bean id="wowan1" factory-bean="factory" factory-method="getPeople1"> <constructor-arg value="0"/> </bean> <bean id="man1" factory-bean="factory" factory-method="getPeople1"> <constructor-arg value="2"/> </bean> 抽象Bean和使用子BeanSpring容器不会初始化抽象Bean(可以没有class属性),子bean可以继承抽象Bean的属性,也可以覆盖
<!-- 抽象bean --> <bean id="ab" abstract="true"> <property name="age" value="10"></property> </bean> <bean id="ab1" parent="ab" class="bean.AbstractBean"> <property name="name" value="ab1"></property> </bean> <bean id="ab2" parent="ab" class="bean.AbstractBean"> <property name="name" value="ab2"></property> </bean> <bean id="ab3" parent="ab" class="bean.AbstractBean"> <property name="name" value="ab3"></property> <property name="age" value="20"></property> </bean>容器中的工厂Bean
上面的工厂Bean是标准工厂Bean,而这个是一种Spring特殊的Bean,需要继承FactoryBean接口,需要实现下面三个方法:
1. T getBean: 实现该方法负责返回该工厂Bean生成的Java实例。
2. Class<?> getObjectType();返回该工厂类实现的实例的实现类
3. boolean isSingleton();
实现该接口的Bean无法像正常的Bean使用,和正常一样设置xml,但是客户端调用的时候,返回的是该工厂的产品,如果需要返回该工厂的实例,需要在id前面加一个&
public class PersonFactory implements FactoryBean { People p = null; @Override public People getObject() throws Exception { if(p==null) p = new People; return p; } @Override public Class getObjectType() { return People.class; } @Override public boolean isSingleton() { return true; } } //返回工厂产品实例 ctx.getBean("people"); //返回工厂实例 ctx.getBean("&people"); 获取Bean本身的ID如果已经获得了实例,但是像获取该实例的id,这就需要这个类实现BeanNameAware接口,我们需要封装一个变量,然后实现接口中的方法,setBeanName方法。
public class Person implements BeanNameAware { private String beanName; public void setBeanName(String beanName) { //这个就是Id this.beanName = beanName; } }
让Bean获取Spring容器
需要让Bean实现一个ApplicationContextAware
public class Person implements ApplicationContextAware { private ApplicationContext context; @Override public void setApplicationContext(ApplicationContext context) throws BeansException { // TODO Auto-generated method stub this.context =context; } }强制初始化:
Spring有个规则,先初始化主调的Bean, 然后初始化依赖Bean。在极端情况下,如果主调的Bean没有初始化,那么就会出现异常。所以就需要强制初始化。
<bean id="peo" class="" depends-on="people"></bean> <bean id="people" class="" ></bean>注入其他Bean的属性值:
PropertyPathFactoryBean用来获取目标bean的属性值(实际上就是使用get方法的返回值),获得的值可以注入给其他对象,也可以直接定义为新的Bean
<bean id="person" class=""> <property name="age" value="30"></property> <property name="son"> <bean class=" "> <property name="age" value="11"></property> </bean> </property> </bean> <bean id="son" class=""> <property name="age"> <bean id="person.son.age" class="org.springframework.beans.factory.config.PropertyPathFactoryBean"></bean> </property> </bean>也可以指定成新的bean,需要指定PropertyPathFactoryBean的两个属性
1. targetBeanName: 用于指定目标Bean, 确定获取哪个Bean的属性值。
2. propertyPath::用于指定属性,确定目标Bean的那一个属性值。
<bean id="person" class=""> <property name="age" value="30"></property> <property name="son"> <bean class=" "> <property name="age" value="11"></property> </bean> </property> </bean> <bean id="son" class="org.springframework.beans.factory.config.PropertyPathFactoryBean"> <property name="targetBeanName" value="person"></property> <property name="propertyPath" value="son"></property> </bean>注入其他Bean的属性值:
通过FieldRetrievingFactoryBean类,可以将其他Bean的Field值,注入给其他的Bean
<bean id="son" class=""> <property name="age"> <bean id="java.sql.Connection.TRANSACTION_SERIALIZABLE" class="org.springframework.beans.factory.config.FieldRetrievingFactoryBean"/> <!--如果是非静态的属性,则需要使用id来定位--> </property> </bean>也可以指定成新的bean,需要指定FieldRetrievingFactoryBean的两个属性
1. targetClass或者targetObject: 指定目标类或者目标对象,如果是静态,则是类,否则这是对象
2. targetField::用于指定目标Field。
<bean id="integer" class="org.springframework.beans.factory.config.PropertyPathFactoryBean"> <property name="targetClass" value="java.sql.Connection"></property> <property name="targetField" value="TRANSACTION_SERIALIZABLE"></property> </bean> 注入其他Bean的方法返回通过MethInvokingFactoryBean,可以指定方法返回值注入生成的Bean
也可以直接返回到一个新的bean中
如果想传入参数 则使用argument
<bean id="teacher"class= "org.springframework.beans.factory.config.MethodInvokingFactoryBean"> <property name="staticMethod"> <value>com.teacher.Teacher.setStudentDao</value> </property> <property name="arguments"> <list> <ref bean="studentDao" /> </list> </property> </bean> <bean id="studentDao" class="com.teacher.StudentDao"> </bean>方法如下 public class Teacher { private static StudentDao studentDao; public static String setStudentDao(StudentDao stuDao) { Teacher.studentDao = stuDao; return "hello world"; } } Bean生命周期:Bean全部属性执行之后指定的行为,有两种
1. 使用init-method属性。
2. 继承InitalizingBean接口。这个优先
Bean销毁之前的行为,有两种
1. 使用destroy-method:
2. 继承DisposableBean接口:这个优先
Spring有良好的扩展性,可以通过Bean的后处理器扩展,可以实现BeanPostProcessor接口,有两个方法
1. Object postProcessBeforeIniialization(Object bean, String name)
2. Object postProcessAfterIniialization(Object bean, String name)
其中,bean是进行后处理的实例,name是该实例的名字。
