Spring4学习笔记二:Bean配置与注入相关

时间:2023-03-08 19:14:15

一:Bean的配置形式

基于XML配置:在src目录下创建 applicationContext.xml  文件,在其中进行配置。

基于注解配置:在创建bean类时,通过注解来注入内容。(这个不好,因为注解也在代码中,而且过于分散)

二:Bean的寻找方式

通过反射来创建bean:通过bean定义时的全类名,用反射机制来寻找bean元数据,创建对象。【因此:Bean类必须至少有一个无参构造函数

id:容器中该bean对象的唯一标识,可以在容器中其他对象中根据id来调用该对象(例如:Factory对象创建时调用数据库连接池对象等),也可以在代码中用ApplicationContext对象.getBean(id)来获取某个bean对象。如果不指定对象,则默认使用类名作为id。

三:Bean对象的赋值——依赖注入的方式

属性注入:在配置bean时,通过<property>标签配置,标签中name指定属性名,value指定属性值,原理是:调用Bean类中的setter方法进行属性赋值。一个标签赋值一个属性。

Spring4学习笔记二:Bean配置与注入相关
<bean id="bean_id" class="www.ygj0930.bean.Bean">
<property name="attr1" value="value1"></property>
<property name="attr2" value="value2"></property>
<property name="attr3" value="value3"></property>
......
</bean>
Spring4学习笔记二:Bean配置与注入相关

构造注入:在配置bean时,通过<constructor-arg>标签为构造函数配置参数值,标签中 value指定参数值,index指定是对应哪个参数,type说明该参数类型。一个标签赋值一个参数。

Spring4学习笔记二:Bean配置与注入相关
<bean id="bean_id" class="www.ygj0930.bean.Bean">
<constructor-arg value="val0" index="0" type="java.lang.XX"></constructor-arg>
<constructor-arg value="val1" index="1" type="java.lang.XX"></constructor-arg>
<constructor-arg value="val2" index="2" type="java.lang.XX"></constructor-arg>
........
</bean>
Spring4学习笔记二:Bean配置与注入相关

属性值、参数值细节:
    1)如果属性值、参数值中包含特殊符号,则需要把属性值、参数值用C-Data括起来,即   <![CDATA[值]]>   形式来配置,都在特殊符号在xml中会被解释,从而报错。

2)属性、参数是基本数据类型、封装数据类型:可以使用  value="字面量"  来注入。

3)属性、参数是其他bean类型:

3.1)引用外部bean:如果该属性、参数变量值是容器中已有bean,则使用  ref="引用的bean_id"  来注入。

3.2)内部bean:如果属性、参数变量是当前bean中内部创建的一个新的bean,则使用<property>标签来配置,在标签之间在内嵌一个<bean>标签,并且以构造注入的方式来创建内部bean。

<bean id="外部bean">
<property name="内部bean的属性名">
<bean class="内部bean的全类名"> //内部bean,没有id,不能被外部其他bean引用
<constructor-arg value="参数值" index="下标" type="参数类型"></constructor-arg>
......
</bean>
</property>
<property name="引用外部bean的属性名" ref="其他bean的ID"></property>
......
</bean>

构造注入的内部bean参数值创建方法同上,标签换成constructor-arg。

4)注入null值

null值的注入有专有的形式,必须按下面格式来写:

<constructor-arg><null/></constructor-arg>

四:级联属性配置

如果一个bean2中引用了另一个bean1作为成员变量,那么我们可以在配置bean2时,通过bean2的成员变量来为引用的bean1赋值或者修改bean1的变量值。

例如:人 的类中有  车,在配置bean时,可以为 person bean引用一个car对象,然后进一步对car对象的属性进行赋值。

【注意:各个类的必须定义各属性的getter和setter方法,否则无法成功注入】

<bean id="person" class="www.ygj0930.bean.Person">
<property name="name" value="ygj"></property>
<property name="age" value="23"></property>
<property name="car" ref="car"></property>
<property name="car.maxSpeed" value="260"></property>
</bean>

五:集合属性赋值

一个Bean类在定义时,包含了集合类型的成员变量的话(list、set等),在配置时,要使用相应的标签逐个赋值:可以用ref引用已有的,也可以用bean标签创建内部bean。

例如:一个人有很多辆车。

<bean id="person" class="www.ygj0930.bean.Person">
<property name="name" value="ygj"></property>
<property name="age" value="23"></property>
<property name="cars"> //集合属性
<list> //对应属性类型来使用标签,还有 set等
<ref bean="car"/> //使用ref标签,指定bean来逐个元素进行赋值
<ref bean="car2"/>
<bean class="全类名"> //创建内部bean来对元素赋值
  <constructor-arg></constructor-arg>
。。。。。。
</bean>
</list>
</property>
</bean>

map类型的属性配置:用map标签,里面元素用entry标签,并且通过entry标签的 key、value或者value-ref来赋值。

<bean id="person" class="www.ygj0930.bean.Person">
<property name="name" value="ygj"></property>
<property name="age" value="23"></property>
<property name="cars">
<map> //使用map标签
<entry key="1" value-ref="car"></entry> //用entry标签来为元素赋值
</map>
</property>
</bean>

Properties类型属性配置:常用于DAO层框架相关bean类配置等。使用props标签包住,里面用prop标签逐个赋值。

<bean id="properties" class="www.ygj0930.bean.TestProperties">
<property name="properties">
<props>
<prop key="name">ygj</prop> //key是属性名,标签之间是属性值
<prop key="password">123</prop>
<prop key="email">11111@111.com</prop>
</props>
</property>
</bean>

六:把集合属性单独配置,供多个bean引用

1:增加命名空间

如下图:Source是xml代码编辑界面,点击  Namespaces  选项卡,切换到命名空间配置界面。

Spring4学习笔记二:Bean配置与注入相关

勾选  util  命名空间。

Spring4学习笔记二:Bean配置与注入相关

2:配置集合内容,指定id。例如:配置map

<util:map id="cars">
<entry key="1" value-ref="car"></entry>
</util:map>

3:在需要引用集合属性值的bean中,通过id来引用集合

    <bean id="person" class="www.ygj0930.bean.Person">
<property name="name" value="ygj"></property>
<property name="age" value="23"></property>
<property name="cars" ref="cars"> //引用上面配置的map
</property>
</bean>

七:使用 p 命名空间,简化bean配置

1:引入 p 命名空间

Spring4学习笔记二:Bean配置与注入相关

2:使用 p:属性名="值"  的形式配置bean属性值。

 <bean id="person" class="www.ygj0930.bean.Person" p:name="ygj" p:age="23" p:cars-ref="cars"></bean> 

八:Bean的自动装配(不推荐使用)——autowire属性

byName:按名自动装配,用于引用其他bean时可以根据bean在容器中的id当前配置bean的属性名  进行匹配,同名则引用。

byType:按类型自动装配,用于根据类型匹配来引用其他bean。缺点:如果某类型的bean不只一个,就会报错。

为什么不推荐使用自动装配:

1:不灵活:如果采用了自动装配配置某个bean,则这个bean的所有 引用  属性都自动装配

2:不清晰:相比于自动装配,逐个配置的话文档看起来更加清晰,知道引用了哪个bean,也方便修改。

九:Bean之间的关系——继承 与 依赖

继承:一个bean可以继承自容器中的另一个bean,并且可以覆盖掉继承来的属性。

<bean id="car1" class="www.ygj0930.bean.Car">
<constructor-arg value="Audi" index="0" type="java.lang.String"></constructor-arg>
<constructor-arg value="ShangHai" type="java.lang.String"></constructor-arg>
<constructor-arg value="1" type="int"></constructor-arg>
</bean>
<bean id="car2" class="www.ygj0930.bean.Car" parent="car1" p:brand="Benz"> //继承car1,覆写个别属性

依赖:一个bean可以依赖于某个bean,在被依赖的bean创建之后才创建自己,否则报错。

  <bean id="person" class="www.ygj0930.bean.Person" p:name="ygj" p:age="23" p:cars-ref="cars" depends-on="car"></bean>  //依赖car

这样的话,需要在car被创建后,才创建person。

十:Bean的作用域

我们可以通过 scope 属性配置bean的作用域。

Spring支持如下5种作用域:

  • singleton:单例模式,在整个Spring IOC容器中,使用singleton定义的Bean将只有一个实例

  • prototype:原型模式,每次通过容器的getBean方法获取prototype定义的Bean时,都将产生一个新的Bean实例

  • request:对于每次HTTP请求,使用request定义的Bean都将产生一个新实例,即每次HTTP请求将会产生不同的Bean实例。只有在Web应用中使用Spring时,该作用域才有效

  • session:对于每次HTTP Session,使用session定义的Bean豆浆产生一个新实例。同样只有在Web应用中使用Spring时,该作用域才有效

  • globalsession:每个全局的HTTP Session,使用session定义的Bean都将产生一个新实例。典型情况下,仅在使用portlet context的时候有效。同样只有在Web应用中使用Spring时,该作用域才有效

  其中比较常用的是singleton和prototype两种作用域。对于singleton作用域的Bean,每次请求该Bean都将获得相同的实例。容器负责跟踪Bean实例的状态,负责维护Bean实例的生命周期行为;如果一个Bean被设置成prototype作用域,程序每次请求该id的Bean,Spring都会新建一个Bean实例,然后返回给程序。在这种情况下,Spring容器仅仅使用new 关键字创建Bean实例,一旦创建成功,容器不在跟踪实例,也不会维护Bean实例的状态。

  如果不指定Bean的作用域,Spring默认使用singleton作用域。Java在创建Java实例时,需要进行内存申请;销毁实例时,需要完成垃圾回收,这些工作都会导致系统开销的增加。因此,prototype作用域Bean的创建、销毁代价比较大。而singleton作用域的Bean实例一旦创建成功,可以重复使用。因此,除非必要,否则尽量避免将Bean被设置成prototype作用域。

十一:引用外部配置文件

对于C3P0等第三方工具的配置,一般使用properties等文件进行各项配置,那么在配置这些bean时怎么获取到里面的具体配置项呢?

Properties文件中各项以 name=value 形式来定义内容,如:

name=Name1
age=21

我们在applicationContext.xml中可以把properties文件加载进容器的上下文中,然后在配置bean时就可以用 ${name} 的形式获取具体的配置值了。

1:添加命名空间

Spring4学习笔记二:Bean配置与注入相关

2:加载文件

注意:配置文件需要与applicationContext.xml位于同一级目录,最好位于src下。

加载格式为:<context:property-placeholder location="classpath:XX.properties>

   获取配置项内容:${配置项名}

 <context:property-placeholder location="classpath:test1.properties"/>
<bean id="person2" class="www.ygj0930.bean.Person" p:name="${name}" p:age="${age}"></bean>

3:加载多个配置文件的命名问题

我们可以通过多个加载语句把多个配置文件加载进context中:

<context:property-placeholder location="classpath:XX.properties"/>
<context:property-placeholder location="classpath:XXX.properties"/>
<context:property-placeholder location="classpath:XXXX.properties"/>
。。。。。。

如果这些文件中存在同名的配置项,例如:都有name配置项,那么我们用 ${name} 获取到的是哪个呢?——第一个导入的文件中的name配置项值。

十二:SpEL表达式

1)语法格式:#{表达式}

2)使用场景:

调用其他bean中的方法和属性:#{bean.func()}、#{bean.attr}

计算表达式的值,为bean的属性进行动态赋值:表达式包括  数学运算+-*/%^;字符串连接;比较运算符><,==,>=,<=;三目运算符:条件 ? value1 :value2 ;逻辑运算符等。

正则表达式的匹配:#{内容 matches '正则表达式'}

静态内容的引用:使用  T(静态类名).变量、方法  来引用,例如:#{T(java.lang.Math).PI}

十三:Bean的生命周期

一般生命周期:Bean的构造函数——bean设置属性(setter函数)——使用bean(代码中getBean,调用bean的一系列属性与方法)

配置了初始化和销毁方法的生命周期:Bean的构造函数——bean设置属性(setter函数)——执行init方法(需要在配置bean时指定)——使用bean(代码中getBean,调用bean的一系列属性与方法)——随着IOC容器的关闭,销毁bean,调用bean的destroy方法(需要在配置bean时指定)

配置了后置处理器的生命周期:Bean的构造函数——bean设置属性(setter函数)——后置处理器执行beforeInit方法——执行init方法(需要在配置bean时指定)——后置处理器执行afterInit方法——使用bean(代码中getBean,调用bean的一系列属性与方法)——随着IOC容器的关闭,销毁bean,调用bean的destroy方法(需要在配置bean时指定)

后置处理器的使用:

1)实现BeanPostProcessor接口,定义后置处理器类;

2)重写其中的两个方法:postProcessAfterInitialization(Object bean, String name)、postProcessBeforeInitialization(Object bean, String name)

在方法中,根据 name 识别bean的名字,执行对应与该bean的校验、过滤、修改工作。

方法的最后,要return bean,把修改后的bean返回。

3)配置后置处理器:在applicationContext.xml中配置一个bean,class为后置处理器类,不需要id。

4)使用:IOC容器自动识别后置处理器,并且在容器中的每个bean初始化方法执行的先后,分别调用postProcessAfterInitialization(Object bean, String name)、postProcessBeforeInitialization(Object bean, String name)这两个方法进行bean的校验和过滤。

十四: