zoukankan      html  css  js  c++  java
  • Spring_3_IOC&DI

    1. 反转控制(IOC Inverse of Control)

    反转控制(IOC Inverse of Control),也称为 转移控制

    • 控制:对于成员变量赋值的控制权;
    • 反转控制:把对于成员变量赋值的控制权,从代码中转移(反转)到 Spring ⼯⼚和配置⽂件中完成。
    • 好处:解耦合;
    • 底层实现:工厂设计模式;

    img1

    2. 依赖注入 (Dependency Injection - DI)

    • 注⼊:通过 Spring 的⼯⼚及配置⽂件,为对象(bean,组件)的成员变量赋值.
    • 依赖注⼊:当⼀个类需要另⼀个类时,就意味着依赖,⼀旦出现依赖,就可以把另⼀个类作为本类的成员变量,最终通过 Spring 配置⽂件进⾏注⼊(赋值)。
    • 好处:解耦合;

    img2

    3. Spring工厂创建复杂对象(3种方式)

    3.1 什么是复杂对象

    img3

    简单对象 : 可以直接通过 new 构造方法创建的对象 复杂对象 :不能直接通过 new 构造方法创建的对象

    3.2 FactoryBean 接口(第一种)

    1. 实现 FactoryBean 接口:实现 getObjectgetObjectTypeisSingleton 方法;

      • getObject():用于书写创建复杂对象时的代码。
      • getObjectType():返回创建的复杂对象的类型。
      • isSingleton():用于决定是否单例。
      public class ConnectionFactoryBean implements FactoryBean<Connection> {
          // 用于书写创建复杂对象时的代码
          @Override
          public Connection getObject() throws Exception {
              Class.forName("com.mysql.jdbc.Driver");
              Connection conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring", "root", "1234");
              return conn;
          }
      
          // 返回创建的复杂对象的类型
          @Override
          public Class<Connection> getObjectType() {
              return Connection.class;
          }
      
          // 是否单例
          @Override
          public boolean isSingleton() {
              return false; // 每一次都创建新的复杂对象
              // return true; // 只创建一次这种类型的复杂对象
          }
      }
      
    2. Spring 配置文件的配置: 如果 class 中指定的类型是 FactoryBean 接⼝的实现类,那么通过 id 值获得的是这个类所创建的复杂对象。 比如下面 class 指定的是 ConnectionFactoryBean,获得的是 Connection 对象。

      <!--class 指定了 ConnectionFactoryBean, 获得的是该类创建的复杂对象 Connection -->
      <bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean"/>
      
    3. FactoryBean 细节 如果就想获得 FactoryBean 类型的对象,加个 & ,ctx.getBean("&conn")

      ConnectionFactoryBean cfb = (ConnectionFactoryBean) ctx.getBean("&conn");
      

      isSingleton 方法返回 true 只会创建⼀个复杂对象,返回 false 每⼀次都会创建新的对象; 需要根据这个对象的特点 ,决定是返回 true(SqlSessionFactory) 还是 false(Connection)

    4. mysql ⾼版本连接创建时,需要制定 SSL 证书,否则会警告;

      Sat May 23 23:18:04 CST 2020 WARN: Establishing SSL connection without server's identity verification is not recommended. 
      According to MySQL 5.5.45+, 5.6.26+ and 5.7.6+ requirements
      SSL connection must be established by default if explicit option
      isn't set. For compliance with existing applications not
      using SSL the verifyServerCertificate property is
      set to 'false'. You need either to explicitly disable
      SSL by setting useSSL=false, or set useSSL=true and provide truststore for server certificate verification.
      

      解决方案:url = jdbc:mysql://localhost:3306/spring?useSSL=false

    5. 加强DI的编程思想: 把 ConnectionFactoryBean 中依赖的 4 个字符串信息 ,通过配置⽂件进行注⼊。

      @Getter@Setter // 提供 get set 方法
      public class ConnectionFactoryBean implements FactoryBean<Connection> {
          // 将依赖的字符串信息变为成员变量, 利用配置文件进行注入。
          private String driverClassName;
          private String url;
          private String username;
          private String password;
          @Override
          public Connection getObject() throws Exception {
              Class.forName(driverClassName);
              Connection conn = DriverManager.getConnection(url, username, password);
              return conn;
          }
          @Override
          public Class<Connection> getObjectType() {
              return Connection.class;
          }
          @Override
          public boolean isSingleton() {
              return false;
          }
      }
      
      <!--体会依赖注入, 好处: 解耦合, 今后要修改连接数据库的信息只需要修改配置文件, 无需改动代码-->
      <bean id="conn" class="com.yusael.factorybean.ConnectionFactoryBean">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/spring?useSSL=false"/>
        <property name="username" value="root"/>
        <property name="password" value="1234"/>
      </bean>
      

    3.3 FactoryBean 实现原理[简易版]

    原理:接口回调。 问题:

    1. 为什么 Spring 规定 FactoryBean 接⼝实现 getObject()?
    2. 为什么 ctx.getBean("conn") 获得的是复杂对象 Connection ⽽非 ConnectionFactoryBean?

    Spring 内部运行流程:

    1. 配置文件中通过 id conn 获得 ConnectionFactoryBean 类的对象 ,进而通过 instanceof 判断出是 FactoryBean 接⼝的实现类;
    2. Spring 按照规定 getObject() —> Connection;
    3. 返回 Connection;

    img4

    FactoryBean 总结:Spring 中⽤于创建复杂对象的⼀种方式,也是 Spring 原⽣提供的,后续 Spring 整合其他框架时会⼤量应⽤ FactoryBean 方式。

    3.4 实例工厂(第二种)

    优势:

    • 避免 Spring 框架的侵⼊;
    • 整合遗留系统;

    开发步骤:

    1. ConnectionFactory 类

      public class ConnectionFactory {
          private String driverClassName;
          private String url;
          private String username;
          private String password;
          public Connection getConnection() {
              Connection conn = null;
              try {
                  Class.forName(driverClassName);
                  conn = DriverManager.getConnection(url, username, password );
              } catch (ClassNotFoundException | SQLException e) {
                  e.printStackTrace();
              }
              return conn;
          }
      }
      
      <!--实例工厂-->
      <!-- 先创建出工厂实例 -->
      <bean id="connFactory" class="com.yusael.factorybean.ConnectionFactory">
        <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
        <property name="url" value="jdbc:mysql://localhost:3306/spring?useSSL=false"/>
        <property name="username" value="root"/>
        <property name="password" value="1234"/>
      </bean>
      <!-- 通过工厂实例里的方法创建复杂对象 -->
      <bean id="conn" factory-bean="connFactory" factory-method="getConnection"/>
      

    3.5 静态工厂(第三种)

    StaticConnectionFactory 类

    public static class staticConnectionFactory {
        public static Connection getConnection() {
            Connection conn = null;
            try {
                Class.forName("com.mysql.jdbc.Driver");
                conn = DriverManager.getConnection("jdbc:mysql://localhost:3306/spring?useSSL=false", "root", "1234");
            } catch (ClassNotFoundException | SQLException e) {
                e.printStackTrace();
            }
            return conn;
        }
    }
    
    <!--静态工厂-->
    <bean id="conn" factory-bean="com.****.****.staticConnerctionFactory" factory-method="getConnection"/>
    

    4. Spring工厂创建对象的总结

    img5

    img6

    4.1 控制 Spring 工厂创建对象的次数

    1. 控制简单对象的创建次数 配置文件中进行配置:

      • sigleton:只会创建⼀次简单对象,默认值;
      • prototype:每⼀次都会创建新的对象;
      <!--控制简单对象创建次数-->
      <bean id="scope" scope="singleton" class="com.yusael.scope.Scope"/>
      
    2. 控制复杂对象的创建次数 如果是 FactoryBean 方式创建的复杂对象:

      public class xxxFactoryBean implements FactoryBean {
          public boolean isSingleton() {
              return true; // 只会创建⼀次
              // return false; // 每⼀次都会创建新的
          }
          // 省略其余实现方法......
      }
      

    如果是实例工厂或者静态工厂,没有 isSingleton ⽅法,与简单对象一样通过 scope 控制。

    4.2 为什么要控制对象的创建次数?

    好处:节省不必要的内存浪费。

    • 什么样的对象只创建⼀次? 重量级的、可以被共用的、线程安全的…

      SqlSessionFactory
      DAO
      Service
      ......
      
    • 什么样的对象每⼀次都要创建新的? 不能被共用的,线程不安全的…

      Connection
      SqlSession | Session
      Struts2 - Action
      ......
      
  • 相关阅读:
    facebook's HipHop for PHP: Move Fast
    使用Linux(CentOS)搭建SVN服务器全攻略
    PHP内置的预定义常量大全
    用PHP纯手工打造会动的多帧GIF图片验证码
    PHP的unset究竟会不会释放内存?
    请远离include_once和require_once
    真希望能夠統一一下接口
    Linux下同步网络时间
    mongo 报connect@src/mongo/shell/mongo.js:251:13错误的解决方式
    spring Aop实现防止重复提交
  • 原文地址:https://www.cnblogs.com/instinct-em/p/13381160.html
Copyright © 2011-2022 走看看