Java框架之Spring01-IOC-bean配置-文件引入-注解装配

JayFighting 2020-01-12

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模块

Java框架之Spring01-IOC-bean配置-文件引入-注解装配

 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个方法

Java框架之Spring01-IOC-bean配置-文件引入-注解装配

<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的实例对象

Java框架之Spring01-IOC-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的名称。

相关推荐