Spring的IOC以及DI

Spring引入

* spring 是什么?

    Spring是分层的 Java SE/EE应用 full-stack 轻量级开源框架,以 IoC(Inverse Of Control:
    反转控制)和 AOP(Aspect Oriented Programming:面向切面编程)为内核,提供了展现层 Spring
    MVC 和持久层 Spring JDBC 以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多
    著名的第三方框架和类库,逐渐成为使用最多的Java EE 企业应用开源框架。

* 程序的耦合
    耦合:程序之间的依赖关系
        包括了类与类之间的依赖,方法之间的依赖

    解耦:降低程序之间的依赖关系,提升程序的灵活性与可扩展性
    实际开发中:应该做到编译期不依赖,运行时才会依赖

    解耦的思路:
        第一步:使用反射来创建对象,从而避免使用new关键字
        第二步:通过读取配置文件来获取要创建的对象全限定类名

* 三层架构代码中存在的问题    
    业务层调用持久层的时候的new dao的接口
    在表现层调用业务层的时候new service的接口

    解决办法: 使用工厂模式解耦(一个创建bean对象的工厂)
        Bean
            1. Bean:在计算机英语中,有可重用组件的含义
            2. JavaBean:用java语言编写的可重用组件
                JavaBean > 实体类
            3. POJO(Plain Ordinary Java Object)即普通Java类
            4. 可以认为POJO就一个不带方法而单纯承载数据实体Bean;EJB是一个包含了POJO的超集、
                包含POJO之外的,具有方法、实现、功能这一群组件。
        方法:
            1. 一个配置文件配置service与dao的内容(xml/properties)
            2. 通过读取配置文件来获取要创建的对象全限定类名,反射加载
            3.优化:使用单例

Spring的IOC(Inversion Of Control)

* IOC:控制反转
    简单理解:让spring来控制对象


* 入门案例:
    1. 导入依赖
    2. 创建配置文件,导入约束,并写入相关的bean标签(配置id与class)
    3. 从容器中获取容器对象
        3.1 先获取容器
        ApplicationContext ac = new ClassPathXmlApplicationContext("配置文件")
        3.2 从容器中根据id或者name获取bean对象
        User user = ac.getBean("user", User.class);
        System.out.println(user);//User{userName='null', age=null, birthday=null}


* ApplicationContext的三个常用的实现类
    ClassPathXmlApplicationContext:它可以加载类路径下的配置文的件,要求配置文件必须在类路径下,不在就加载不了
    FileSystemXmlApplicationContext:它可以加载磁盘任意路径下的配置文件    (必须有访问权限)
    AnnotationConfigApplicationContext:它是用于读取注解的配置

* 核心容器的两个接口引发出的问题
    ApplicationContext:(单例对象试用),多使用此接口
        它在构建核心容器的时候,创建对象采取的是立即加载的方式,也就是说,只要一读取完配置文件就马上
        就创建配置文件中的配置的对象
    BeanFactory:(多例对象适用)
        它在构建核心容器的时候,采取的策略是延迟加载的方式,也就是说,什么时候根据id获取对象了,什么时候才会
        真正的创建对象

* Spring对bean的管理细节
    1. 创建bean的三种方式:

        1.1 第一种方式:使用默认构造函数创建
            在spring的配置文件中使用bean标签,配置id与class属性之后,且没有其他属性和标签时
            采用的就是默认构造函数创建bean对象,测试如果类中没有默认构造函数,则对象无法创建
            <bean id="service" class="com.qin.service.impl.IUserServiceImpl"></bean>
        1.2 第二种方式:使用普通工厂中的方法来创建对象(使用某个类中的方法创建对象,并存入spring容器)
            <bean id="instanceFactory" class="com.qin.factory.instanceFactory"></bean>(要先有对象才能调用方法)
            <bean id="serviceFactory" class="com.qin.factory.instanceFactory" factory-method="getAccount">
            </bean>
        1.3 第三种方式:使用工厂中的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
            <bean id="serviceFactory" class="com.qin.factory.StaticFactory" factory-method="getAccount">
            </bean>
            静态方法是不需要对象的,是可以通过全类名调用的,所以不需要对象

    2. bean的作用范围
            bean标签的scope属性
                作用:用于指定bean 的作用范围
                取值:
                    singleton:单例的(默认值)
                    prototype:多例的
                    request:作用于wen应用的请求范围
                    session:作用于wen应用的会话范围
                    global-session:作用于集群环境的会话范围(全局会话范围),当不是集群的时候,相当于session
    3. bean对象的生命周期
        单例对象:
            出生:当容器创建时候对象出生
            存在:只要容器在,对象就一直活着
            死亡:容器销毁,对象消亡
            总结:单例对象的生命周期和容器相同
        多例对象:
            出生:当使用对象时,spring框架为我们创建
            存在:对象还在使用的过程中就一直存活
            死亡:当对象长时间不用,且没有别的对象引用时,由java的垃圾回收器回收

Spring依赖注入:DI(依赖关系的维护)

    * 能注入的数据:三类
            1. 基本类型和string :
            2. 其他bean类型(在配置文件中或者注解配置过的bean)
            3. 复杂类型/集合类型
    * 注入的方式:有三种
            1. 使用构造函数提供(不常用)
                使用的标签:constructor-arg
                标签出现的位置:bean标签的内部
                标签的属性:
                    type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型
                    index:用于指定要注入的数据给构造函数中指定索引位置的参数赋值,索引的位置是从0开始的
                    name:用于指定给构造函数中指定名称的参数赋值(常用)
                    =============以上三个用于指定给构造函数中的哪个参数赋值==================
                    value:用于提供基本类型和string类型的数据
                    ref:用于指定其他的bean类型数据.他指的就是在spring的核心容器中出现过的bean对象

                    优势:
                        在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功.
                    弊端:
                        改变了bean对象的实例化方式,使我们在创建对象时,如果不使用这些数据也必须提供.

            2. 使用set方法提供(常用)
                * 首先必须有set方法
                涉及的标签:property
                出现的位置:bean标签的内部
                标签的属性:
                    name:用于指定注入时调用的set方法名称
                    value:用于提供基本类型和string类型的数据
                    ref:用于指定其他的bean类型数据.他指的就是在spring的核心容器中出现过的bean对象

                优势:创建对象时没有明确的限制,可以直接使用默认构造函数
                弊端:如果某个成员必须有值,则获取对象是有可能set方法没有执行

            3. 使用注解提供

    * 示例:
        1. 构造函数注入
            <bean id="userDao" class="com.qin.dao.impl.IUserDaoImpl">
                <constructor-arg name="userName" value="sakura"></constructor-arg>
                <constructor-arg name="age" value="18"></constructor-arg>
                <constructor-arg name="birthday" ref="date"></constructor-arg>
            </bean>

            <bean id="date" class="java.util.Date"></bean>

        2. 使用set方法注入
             <bean id="userDao1" class="com.qin.dao.impl.IUserDaoImpl1">
                <property name="age" value="19"></property>
                <property name="userName" value="tom"></property>
                <property name="birthday" ref="date"></property>
            </bean>

            <bean id="date" class="java.util.Date"></bean>

        3. 对集合属性进行set方式注入
            <bean id="userDao2" class="com.qin.dao.impl.IUserDaoImpl2">
                <property name="myStrs">
                    <array>
                    <value>string</value>
                    <value>string</value>
                    <value>string</value>
                    </array>
                </property>

                <property name="myList">
                    <list>
                        <value>list</value>
                        <value>list</value>
                        <value>list</value>
                    </list>
                </property>
                <property name="mySet">
                    <set>
                        <value>set</value>
                        <value>set</value>
                        <value>set</value>
                    </set>
                </property>
                <property name="myMap">
                    <map>
                        <entry key="map1">
                            <value>map1</value>
                        </entry>
                        <entry key="map2" value="map2"></entry>
                        <entry key="map3" value="map3"></entry>
                    </map>
                </property>
                <property name="myProps">
                    <props>
                        <prop key="prop">123</prop>
                        <prop key="prop">123</prop>
                        <prop key="prop">123</prop>
                    </props>
                </property>
            </bean>

Spring的基于注解开发

* IOC在XML的配置:<bean id="accountService" class="com.qin.service.impl.AccountService" scope=""
        init-method="" destory-method="">
        <property name="" value="" |ref=""></property>
         </bean>

* 注解分类:
    1. 用于创建对象的
        它们的作用就和xml配置中编写一个<bean>标签实现的功能是一样的

        @Component:
            作用:
                用于把当前类对象存入Spring容器中
            属性:
                value:用于指定bean中的id.当我们不写时,他的默认值是当前类名的,且首字母小写

        @Controller:一般用在表现层
        @Service:一般用在业务层
        @Repository:一般用在持久层
        以上三个注解的作用和属性与Component是一模一样的
        他们三个是spring为我们提供明确的三层使用的注解,使我们的三层对象更加清晰

        在xml文件的配置:
            <context:component-scan base-package="com.qin"></context:component-scan>
            作用:告知spring在创建容器时要扫描的包,配置所需要的标签不是beans约束中,而是在context
            的名称空间和约束中

            约束配置:
            <?xml version="1.0" encoding="UTF-8"?>
            <beans xmlns="http://www.springframework.org/schema/beans"
                xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
                xmlns:context="http://www.springframework.org/schema/context"
                xsi:schemaLocation="http://www.springframework.org/schema/beans
                http://www.springframework.org/schema/beans/spring-beans.xsd
                http://www.springframework.org/schema/context
                http://www.springframework.org/schema/context/spring-context.xsd">

    2. 用于注入数据的
        它们的作用就和在xml配置文件中的bean标签中写一个<property>标签的作用是一样的

        @Autowired
            作用:(自动按照类型注入)    
                如果ioc容器中有唯一的一个bean对象类型和要注入的变量类型匹配,就可以注入成功
                如果ioc容器中没有任何bean类型和要注入的变量类型一致,将会注入失败
                如果ioc容器中有多个bean类型和要注入的变量类型一致,会先根据类型寻找,再根据变量名找
            出现位置:
                变量上,方法上(在set方法上也可以使用);
            细节:
                在使用注解注入时,set方法就不是必须的
        @Qualifier
            * 配合@Autowired使用
            * 也可以存在于参数上,配合指定构造中要传递的bean对象
            作用:
                在按照类中注入的基础之上再按照名称注入.他在给类成员注入时不能单独使用,但是在给方法参数注入时可以
            属性:
                value:用于指定注入bean的id
        @Resource
            * jdk1.9不能使用
            作用:
                直接按照bean的id注入,它可以单独使用
            属性:
                name:用于指定bean 的id

        * 以上三个注入都只能注入其他bean类型,而基本类型和string类型无法使用上述注解实现,集合类型也可以用Autowired实现

        @Value
            作用:
                用于注入基本类型和string类型的数据
            属性:
                value:用于指定数据的值.他可以使用spring和spEL(也就是spring的el表达式)
                spEL的写法:#{表达式}

                #{}:去容器中找id为花括号内容对应的值
                ${}:去配置文件中读取花括号对应的值
    3. 用于改变作用范围的
        它们的作用就和在xml配置文件中的bean标签中使用scope属性实现的功能是一样的

        @Scope
            作用:
                用于指定bean的作用范围
            属性:
                value:指定范围的取值.常用取值:singleton prototype

    4. 和生命周期相关
        它们的作用就和在xml配置文件中的bean标签中使用init-method和destory-method的作用是一样的

        @PreDestroy
            作用:
                用于指定销毁方法
        @PostConstruct
            作用: 
                用于指定初始化方法


* spring的新注解(可以使用纯注解的形式开发)
    * 创建一个类.替换核心配置文件
    @Configuration
        作用:
            指定当前类是一个配置类
        细节:
            当配置类作为AnnotationConfigApplicationContext对象创建的参数时,可以不写此注解
            但不是任何时候都可以不写(把注解文件分开写,另一个没有读的配置文件必须写)
    @ConponentScan
        作用:
            用于通过注解指定spring在创建容器时要扫面的包
        属性:
            value:它和basePackage的作用是一样的,都是用于指定创建容器时要扫面的包
            我们使用次注解就等同于在xml中配置了:
            <context:component-scan base-package="com.qin"></context:component-scan>


    @Bean
        作用:
            用于把当前方法的的返回值作为bean对象存入spring的ioc容器中
        属性:
            name:用于指定bean的id,当不写时,默认值是当前方法的名称
        细节:
            当我们使用注解来配置方法时,如果方法有参数,spring框架会去容器中查找有没有可用的bean对象,查找的方式和
            @Autowired注解的作用一样的

    @Import
        作用:
            用于导入其他的配置类
        属性:
            value:用于指定其他配置类的字节码文件
            当我们使用import的注解之后,有import注解的类就是父配置类,而导入的都是子配置类

    @PropertySource
        作用:
            用于指定properties文件的位置
        属性:
            value:指定文件的名称以及文件的路径
                关键字:classpath,表示类路径下
        例子:
            @PropertySource("classpath:JdbcConfig.properties")


    *没有xml文件时测试就要修改ApplicationContext的创建方式
    ApplicationContext ac = new AnnotationConfigApplicationContext("被@Configuration注解过的类".class)

抽取配置数据库文件

1.抽取文件:
jdbcConfig.properties
    jdbc.driver=com.mysql.jdbc.Driver
    jdbc.url=jdbc:mysql://localhost:3306/spring
    jdbc.username=root
    jdbc.password=root

2. 在Spring的配置文件中配置:
    <context:property-placeholder location="jdbcConfig.properties"></context:property-placeholder>
    使用注解的配置为:
        @PropertySource("classpath:jdbcConfig.properties")


3. 在原来的配置文件中使用Spring的el表达式
    ${}来表示
    <property name="driverClass" value="${jdbc.driver}"></property>
    <property name="jdbcUrl" value="${jdbc.url}"></property>
    <property name="user" value="${jdbc.username}"></property>
    <property name="password" value="${jdbc.password"></property>

猜你喜欢

转载自blog.csdn.net/qq_35472880/article/details/83314454