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

 

一: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方法进行属性赋值。一个标签赋值一个属性。

复制代码
<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>
复制代码

 

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

复制代码
<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>
复制代码

     

    属性值、参数值细节:
    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  选项卡,切换到命名空间配置界面。

勾选  util  命名空间。

 

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 命名空间

 

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:添加命名空间

   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的校验和过滤。

 

十四:

 

  

 

转载于:https://www.cnblogs.com/ygj0930/p/8257608.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值