zoukankan      html  css  js  c++  java
  • Java框架之Spring 01-IOC-bean配置-文件引入-注解装配

    Spring

     框架,即framework。是对特定应用领域中的应用系统的部分设计和实现的整体结构。就相当于让别人帮你完成一些基础工作,它可以处理系统很多细节问题,而且框架一般是成熟,稳健的。

     Spring概述

       Spring是一个IOC(DI)和AOP容器框架

       Spring的优良特性

         ①   非侵入式:基于Spring开发的应用中的对象可以不依赖于Spring的API

         ②   依赖注入:DI——Dependency Injection,反转控制(IOC)最经典的实现。

         ③   面向切面编程:Aspect Oriented Programming——AOP

         ④   容器:Spring是一个容器,因为它包含并且管理应用对象的生命周期

         ⑤   组件化:Spring实现了使用简单的组件配置组合成一个复杂的应用。在 Spring 中可以使用XML和Java注解组合这些对象。

         ⑥  一站式:在IOC和AOP的基础上可以整合各种企业应用的开源框架和优秀的第三方类库(实际上Spring 自身也提供了表述层的SpringMVC和持久层的Spring JDBC)。

     Spring模块

     IOC和DI

     IOC(Inversion of Control):反转控制

       反转了资源的获取方向——改由容器主动的将资源推送给需要的组件,开发人员不需要知道容器是如何创建资源对象的,只需要提供接收资源的方式即可。这种行为也称为查找的被动形式。

     DI(Dependency Injection):依赖注入

       即组件以一些预先定义好的方式(例如:setter 方法)接受来自于容器的资源注入。

       总结: IOC 就是一种反转控制的思想, 而DI是对IOC的一种具体实现。

    IOC容器在Spring中的实现

      导入Spring框架jar包

      创建配置文件,常用文件名:applicationContext.xml或beans.xml

    • Spring中有IOC思想,  IOC思想必须基于 IOC容器来完成, 而IOC容器在最底层实质上就是一个对象工厂

        1)在通过IOC容器读取Bean的实例之前,需要先将IOC容器本身实例化

        2)Spring提供了IOC容器的两种实现方式

      ① BeanFactory:IOC容器的基本实现,是Spring内部的基础设施,是面向Spring本身的,不是提供给开发人员使用的。

      ② ApplicationContext:BeanFactory的子接口,提供了更多高级特性。面向Spring的使用者,几乎所有场合都使用ApplicationContext而不是底层的BeanFactory。

    ApplicationContext的主要实现类

      ClassPathXmlApplicationContext:对应类路径下的XML格式的配置文件

      FileSystemXmlApplicationContext:对应文件系统中的XML格式的配置文件

      在初始化时就创建单例的bean,也可以通过配置的方式指定创建的Bean是多实例的。

    ConfigurableApplicationContext

          是ApplicationContext的子接口,包含一些扩展方法

          refresh()和close()让ApplicationContext具有启动、关闭和刷新上下文的能力。所以要关闭ApplicationContext需要new此接口的对象调用close()方法

    WebApplicationContext

           专门为WEB应用而准备的,它允许从相对于WEB根目录的路径中完成初始化工作

      从IOC容器中获取bean,推荐同时指定bean的id值和类型

    //通过在XML文件中配置的id及class类型获取对象
    HelloWorld helloWorld = cxt.getBean("helloWorld",HelloWorld.class);

    bean标签

      bean标签:将bean装配到springIOC容器中 
      bean标签中属性
         id:对象唯一标识(可以不写,如果书写必须是唯一值)
           class:装配bean的全类名

      bean子标签

        property:为对象中的属性赋值
          name:属性名
          value:属性值

        constructor-arg:通过构造器赋值 

    给bean的属性赋值

    1. 通过bean的setXxx()方法赋值

    2. 通过bean的构造器赋值,注意:如果构造器中参数类型兼容,可能出现错误赋值情况。

    <bean id="student" class="com.bean.Student01">
            <property name="name" value="小明"></property>
            <constructor-arg name="age" value="18"></constructor-arg>
    </bean>

    3. p名称空间

    <bean id="student" class="com.bean.Student01"
            p:name="小明" 
            p:age="18">
    </bean>

    属性可使用的值

    1. 字面量

          基本数据类型及其封装类、String等类型都可以采取字面值注入的方式

          若字面值中包含特殊字符,可以使用<![CDATA[]]>把字面值包裹起来或转义字符

    2. null值

    <property name= "bookName">
             <null/>
    </property>

    3. 给bean的级联属性赋值

      设置级联属性后会修改原属性值,一般不使用

    4. 外部已声明的bean、引用其他的bean:此时value已经满足不了需求了要用ref属性

    <bean id="school" class="com.bean.School">
            <property name="stus" ref="stu"></property>
    </bean>

    5. 内部bean

      当bean实例仅仅给一个特定的属性使用时,可以将其声明为内部bean。内部bean声明直接包含在<property>或<constructor-arg>元素里,不需要设置任何id或name属性

      内部bean不能使用在任何其他地方,即不能在容器中直接获取内部bean

    为bean注入集合属性

    数组和List

      需要指定<list>标签,在标签里包含一些元素。这些标签可以通过<value>指定简单的常量值,通过<ref>指定对其他Bean的引用。通过<bean>指定内置bean。通过<null/>指定空元素。甚至可以内嵌其他集合。

      数组的定义和List一样,都使用<list>元素。

      配置java.util.Set需要使用<set>标签,定义的方法与List一样。

    <bean id="shop" class="com.spring.bean.Shop" >
          <property name= "bookList">
               <!-- 以bean的引用为值的List集合 -->
               <list>
                   <ref bean= "book01"/>
                   <ref bean= "book02"/>
               </list>
           </property>
    </bean >

    Map

      通过<map>标签定义,<map>标签里可以使用多个<entry>作为子标签。每个条目包含一个键和一个值。

            必须在<key>标签里定义键,因为键和值的类型没有限制,所以可以自由地为它们指定<value>、<ref>、<bean>或<null/>元素。

    <bean id="cup" class="com.spring.bean.Cup">
        <property name="bookMap">
            <map>
               <entry key="book" value-ref="bookMap"></entry>
                 <entry>
                      <key>
                          <value>bookKey01</value>
                      </key>
                      <ref bean="book01"/>
                  </entry>
            </map>
        </property>
    </bean>                

    集合类型的bean

      将集合bean的配置提取到外面,可供其他bean引用,实现重用

     <util:list id="schoolList">
            <ref bean="stu"></ref>
    </util:list>

    FactoryBean

      如果需要程序员参与创建bean的过程之中,使用FactoryBean

      工厂bean跟普通bean不同,其返回的对象不是指定类的一个实例,其返回的是该工厂bean的getObject方法所返回的对象。

      工厂bean必须实现org.springframework.beans.factory.FactoryBean接口,并重写3个方法

    <bean id="schoolFactory" class="com.factoryBeanImpl.SchoolFactory">
    </bean>

    bean的作用域

      可以在<bean>元素的scope属性里设置bean的作用域,以决定这个bean是单实例的还是多实例的。

      singleton,是所有bean的默认作用域。注意:工厂bean是通过isSingleton()方法设置是否单例的

      当bean的作用域为单例时,Spring会在IOC容器对象创建时就创建bean的对象实例

      而当bean的作用域为prototype时,IOC容器在获取bean的实例时创建bean的实例对象

    bean的生命周期

          在配置bean时,通过init-method和destroy-method 属性为bean指定初始化和销毁方法

          Spring IOC容器对bean的生命周期进行管理的过程:

             ① 通过构造器或工厂方法创建bean实例

             ② 为bean的属性设置值和对其他bean的引用

             ③ 调用bean的初始化方法

             ④  bean可以使用了

             ⑤ 当容器关闭时,调用bean的销毁方法

    <bean id="stu" class="com.bean.Student" init-method="init" destroy-method="destroy">
    </bean>

       bean的后置处理器

         ① bean后置处理器允许在调用初始化方法前后对bean进行额外的处理

                   ② bean后置处理器对IOC容器里的所有bean实例逐一处理,而非单一实例。

         ③ bean后置处理器需要实现接口:org.springframework.beans.factory.config.BeanPostProcessor。

          在初始化方法被调用前后,Spring将把每个bean实例分别传递给上述接口的以下两个方法:

          ●postProcessBeforeInitialization(Object bean, String beanId):初始化之前执行

          ●postProcessAfterInitialization(Object, String):初始化之后执行

       注意

         参数bean:IOC容器中创建的对象

         参数beanId:IOC容器中创建对象的beanId

    引用外部文件

      将一部分信息提取到bean配置文件的外部,以properties格式的属性文件保存起来,同时在bean的配置文件中引用properties属性文件中的内容,从而实现一部分属性值在发生变化时仅修改properties属性文件即可。

    1. 创建properties属性文件

    jdbc.username=root
    jdbc.password=12345
    jdbc.driverClass=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://localhost:3306/test

    2. 引入context名称空间

    3.指定properties属性文件的位置

    <context:property-placeholder location="classpath:jdbc.properties"></context:property-placeholder>

    classpath: 引入当前项目中类路径下的资源文件
    classpath*: 引入所项目中类路径下的资源文件

    4.从properties属性文件中引入属性值

    <bean id="dataSource" class="com.alibaba.druid.pool.DruidDataSource"
            p:username="${jdbc.username}"
            p:password="${jdbc.password}"
            p:driverClassName="${jdbc.driverClass}"
            p:url="${jdbc.url}">
    </bean>

    自动装配

          手动装配:在XML配置文件中以value或ref的方式明确指定属性值都是手动装配。

          自动装配:根据指定的装配规则,不需要明确指定,Spring容器会自动将匹配的属性值注入bean中。

          注意:自动装配属性的数据类型,只能是[非字面量]值。[字面量]值不能自动装配。即基本数据类型(包装类)+String类型都不可自动装配

     装配方式

    1.  根据类型自动装配:将类型匹配的bean作为属性注入到另一个bean中。当有多个与目标bean类型一致将报错
    2.  根据名称自动装配:必须将目标bean的名称和属性名设置的完全相同
    3. 通过构造器自动装配:当bean中存在多个构造器时,此种自动装配方式将会很复杂。不推荐使用。

     基于xml,自动装配(不推荐)

       在bean中添加autowire="byName|byType"

    <bean id="student" class="com.springdemo.autowired.Student" autowire="byType"></bean>

      byName:通过类中的属性名与bean中id(IOC容器中的id匹配)。
      * 如果数值一致,匹配成功。 如果不一致,装配失败(不会报错,装配null值)

      byType:通过类中的属性类型与bean中的class匹配
      * 如果一致,匹配成功。
      * 如果未找到匹配类型,装配失败(装配null值)
      * 如果匹配到多个兼容类型(父子关系:装配失败,结果会报错)

     基于注解,自动装配bean

       需在XML文档中先添加扫描组件标签,指定需被装配bean的package

    <context:component-scan base-package="com.bookStore" use-default-filters="true"></context:component-scan>

      base-package:Spring容器会扫描这个基类包及其子包中的所有类。当需要扫描多个包时可以使用逗号分隔。

      如果仅希望扫描特定的类而非基包下的所有类,可使用resource-pattern属性过滤特定的类

      use-default-filters="true":默认组件扫描(扫描当前base-package下的包及其子包),false:不扫描...

    通过子标签控制包含与排除

      <context:include-filter>包含扫描,扫描指定匹配规则下的包及其子包

        注意:通过将use-default-filters属性设置为false,禁用默认过滤器,然后扫描的就只是include-filter中的规则指定组件了    

      <context:exclude-filter>排除扫描,子节点表示要排除在外的目标类

        注意:将use-default-filters属性设置为true或默认

    过滤表达式,指定类型

    类别

    示例

    说明

    annotation

    com.XxxAnnotation

    过滤所有标注了XxxAnnotation的类。这个规则根据目标组件是否标注了指定类型的注解进行过滤。

    assignable

    com.BaseXxx

    过滤所有BaseXxx类的子类。这个规则根据目标组件是否是指定类型的子类的方式进行过滤。

    aspectj

    com.*Service+

    所有类名是以Service结束的,或这样的类的子类。这个规则根据AspectJ表达式进行过滤。

    regex

    com.anno.*

    所有com.anno包下的类。这个规则根据正则表达式匹配到的类名进行过滤。

    custom

    com.XxxTypeFilter

    使用XxxTypeFilter类通过编码的方式自定义过滤规则。该类必须实现org.springframework.core.type.filter.TypeFilter接口

    4个注解:

      1) 普通组件:@Component

      2) 表述层控制器组件:@Controller

      3) 业务逻辑层组件:@Service

      4) 持久化层组件:@Repository

      组件命名规则

             ①默认情况:使用组件的简单类名首字母小写作为bean的id

        ②使用组件注解的value属性指定bean的id:  @Component(value="指定id名")

    自动装配bean中的属性

    实现原理

      在指定要扫描的包时,<context:component-scan> 元素会自动注册一个bean的后置处 理器:AutowiredAnnotationBeanPostProcessor的实例。该后置处理器可以自动装配标记 了@Autowired、@Resource或@Inject注解的属性。

    @Autowired

      注入方式:既不是set注入,也不是构造注入。本质:是通过反射注入。

      自动装配规则 

        优先使用byType进行装配,如果能唯一匹配,则装配成功。
        如果匹配到多个兼容类型的bean,再照byName方式匹配(进行唯一筛选)
        如果通过byName唯一确定bean,则装配成功,否则装配失败。

      构造器、普通字段(即使是非public)、一切具有参数的方法都可以使用@Autowired注解

      若某一属性允许不被装配,可以设置@Autowired注解的required属性为 false

        required:默认值是true,必须装配该bean

        required值为false时,如果IOC容器中存在该bean,则装配。如果没有,则不装配。

      @Autowired注解也可以应用在数组类型的属性上

      @Autowired注解也可以应用在集合属性上,此时Spring读取该集合的类型信息,然后自动装配所有与之兼容的bean。

      @Autowired注解用在java.util.Map上时,若该Map的键值为String,那么 Spring将自动装配与值类型兼容的bean作为值,并以bean的id值作为键。

    @Qualifier

      必要时,可以组合使用@Qualifier(value="userDaoMyBatisImpl")注解指定beanId名

      Spring甚至允许在方法形参上标注@Qualifiter注解以指定注入bean的名称。

  • 相关阅读:
    Java ConcurrentModificationException 异常分析与解决方案
    Kafka剖析(一):Kafka背景及架构介绍
    Linux下更改oracle客户端字符集和服务端字符集
    storm-kafka源码走读之KafkaSpout
    kafka系列之(3)——Coordinator与offset管理和Consumer Rebalance
    Kafka源码深度解析-序列7 -Consumer -coordinator协议与heartbeat实现原理
    apache kafka系列之在zookeeper中存储结构
    Java transient关键字使用小记
    Kafka设计解析(一)- Kafka背景及架构介绍
    JavaScript-如何获取页面元素对象,元素id
  • 原文地址:https://www.cnblogs.com/Open-ing/p/12158621.html
Copyright © 2011-2022 走看看