Springフレームワークとソースコード(2)---Spring IoCの高度なアプリケーションとソースコードの解析

春のIOCアプリケーション

セクション 1 Spring IoC の基礎

Spring フレームワークでの IOC 実装、Bean を解析するいくつかの方法

ここに画像の説明を挿入します

1.1 BeanFactory と ApplicationContext の違い

BeanFactorySpringフレームワーク内の IoC コンテナの最上位インターフェイスです、これはいくつかの基本的な機能を定義し、いくつかの基本的な仕様を定義するためにのみ使用されますが、ApplicationContextそのサブインターフェースであるため、ApplicationContextBeanFactory提供されているすべての機能が備わっています 通常、基本コンテナとBeanFactory呼ばれるものは、国際化サポートやリソース アクセス ( ) など、より多くの機能を備えたコンテナへの高レベルのインターフェイスです。SpringIOCApplicationContextBeanFactoryxml,java配置类
ここに画像の説明を挿入します

IoCコンテナの起動方法
  • Java環境でIoCコンテナを起動する

ClassPathXmlApplicationContext:クラスのルート パスから設定ファイルをロードします (推奨)
FileSystemXmlApplicationContext :ディスク パスから設定ファイルをロードします
AnnotationConfigApplicationContext:純粋なアノテーション モードで Spring コンテナを起動します

ClassPathXmlApplicationContext と FileSystemXmlApplicationContext のデモ

applicationContext.xml を定義する

<?xml version="1.0" encoding="UTF-8"?>
<!--跟标签beans,里面配置一个又一个的bean子标签,每一个bean子标签都代表一个类的配置-->
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="
        http://www.springframework.org/schema/beans
        https://www.springframework.org/schema/beans/spring-beans.xsd
">
    <!--bean
        作用:
            声明类交给spring容器
        属性:
            id: 唯一标识
            class:全路径限定名称
        细节:
            默认使用无参构造函数实例化-->
    <bean id="accountDao" class="com.lagou.edu.dao.impl.JdbcAccountDaoImpl">
        <property name="ConnectionUtils" ref="connectionUtils"/>
    </bean>
    <bean id="transferService" class="com.lagou.edu.service.impl.TransferServiceImpl">
        <!--set+ name 之后锁定到传值的set方法了,通过反射技术可以调用该方法传入对应的值-->
        <property name="AccountDao" ref="accountDao"></property>
    </bean>

    <!--配置新增的三个Bean-->
    <bean id="connectionUtils" class="com.lagou.edu.utils.ConnectionUtils"></bean>

    <!--事务管理器-->
    <bean id="transactionManager" class="com.lagou.edu.utils.TransactionManager">
        <property name="ConnectionUtils" ref="connectionUtils"/>
    </bean>

    <!--代理对象工厂-->
    <bean id="proxyFactory" class="com.lagou.edu.factory.ProxyFactory">
        <property name="TransactionManager" ref="transactionManager"/>
    </bean>
</beans>

    @Test
    public void testIoC() throws Exception {
    
    
        // 通过读取classpath下的xml文件来启动容器(xml模式SE应用下推荐)
        ClassPathXmlApplicationContext applicationContext = new ClassPathXmlApplicationContext("classpath:applicationContext.xml");
        // 不推荐使用
        //ApplicationContext applicationContext1 = new FileSystemXmlApplicationContext("文件系统的绝对路径");
//        ApplicationContext applicationContext = new FileSystemXmlApplicationContext("F:\\outWork\\00-kaiyuan\\06-spring\\spring\\lagou-transfer-ioc-xml\\src\\main\\resources\\applicationContext.xml");
        AccountDao accountDao = (AccountDao) applicationContext.getBean("accountDao");
        System.out.println("accountDao:" + accountDao);
        applicationContext.close();
    }

ここに画像の説明を挿入します
まとめ

    1、工厂类:ApplicationContext
    2、工厂配置:
        文件类型:xml
        方式:
       <!--bean
        作用:
            声明类交给spring容器
        属性:
            id: 唯一标识
            class:全路径限定名称
        细节:
            默认使用无参构造函数实例化-->
    <bean id="accountDao" class="com.lagou.edu.dao.impl.JdbcAccountDaoImpl">
        <property name="ConnectionUtils" ref="connectionUtils"/>
    </bean>
    3、工厂加载配置:
        ApplicationContext通过ClassPathXmlApplicationContext加载applicationContext.xml配置;
        或者通过FileSystemXmlApplicationContext加载applicationContext.xml配置;
    4、工厂获得bean:
        ApplicationContext使用getBean()方法,用于根据bean的名称获取实例化对象

Web 環境で IoC コンテナを開始します (リスナーを介して XML をロードします)。

1. XMLからコンテナを起動する

  • Spring-web に依存関係
    ContextLoaderListener を追加するため、依存関係を追加する必要があります
<!--引入spring web功能-->
    <dependency>
      <groupId>org.springframework</groupId>
      <artifactId>spring-web</artifactId>
      <version>5.1.12.RELEASE</version>
    </dependency>
  • web.xmlを構成する
<!DOCTYPE web-app PUBLIC
 "-//Sun Microsystems, Inc.//DTD Web Application 2.3//EN"
 "http://java.sun.com/dtd/web-app_2_3.dtd" >

<web-app>
  <display-name>Archetype Created Web Application</display-name>
  <!--配置Spring ioc容器的配置⽂件-->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:applicationContext.xml</param-value>
  </context-param>
  <!--使⽤监听器启动Spring的IOC容器-->
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>
</web-app>

  • オブジェクトの取得
    private TransferService transferService = null;
    //通过webApplicationContext来获取对象实例
    @Override
    public void init() throws ServletException {
    
    
        WebApplicationContext webApplicationContext = WebApplicationContextUtils.getWebApplicationContext(this.getServletContext());
         transferService = (TransferService) .getBean("transferService");

    }
  • 分析する

ここに画像の説明を挿入します

2.構成クラスからコンテナを起動します

セクション2 beanタグの詳細説明【ポイント】

1.Beanタグの機能

bean作用:  
	用于配置对象让spring 来创建的。 
    【细节】
	默认情况下调用类的无参构造函数。

2. Beanタグの基本属性

属性 説明する
ID Bean の一意の識別名
クラス 実装クラスの完全修飾名
名前 Bean名
※複数のエイリアスは「,」で区切ります
※BeanとBeanエイリアスは重複できません
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd">
    
    <bean id="accountDao" name="accountDao2,accountDao3" class="com.spring.dao.impl.AccountDaoImpl"></bean>

    <bean id="accountService"  class="com.spring.service.impl.AccountServiceImpl"></bean>

</beans>

3.Beanタグの範囲

属性 説明する
範囲 オブジェクトのスコープを指定します。
* singleton[デフォルト]: 単一インスタンス、すべてのリクエストは 1 つのオブジェクトで処理されます
* prototype: 複数インスタンス、各リクエストは新しいオブジェクトで処理されます
* request: WEB プロジェクトでは、オブジェクトはリクエスト フィールドに格納されます
* session: WEB プロジェクトでは、オブジェクトはセッションドメインに格納されます
global session:WEBプロジェクトではクラスタ環境で適用されます クラスタ環境がない場合はセッションと同等

グローバルセッション図

ここに画像の説明を挿入します

【1】思考
单例、多例他们分别在什么场景中使用?他们有什么区别?
spring默认单例,不需要修改,不要随意定义成员变量。
多例:资源共用
【2】目標
1、掌握scope的单例、多例的配置
2、掌握单例和多例的区别
【3】Beanスコープインスタンス (6)
步骤:
	1、改造ClientController多次获得对象
	2、装配bean到spring的IOC容器中,修改bean标签中scope的作用域
	3、观察不同作用域下获得的对象内存地址是否一致
【3.1】プロジェクトの作成

クライアントコントローラー


/**
 * @Description:测试
 */
public class ClientController {
    
    
	

    @Test
    public void saveAccount() {
    
    

        /**
         * Spring-IOC容器:ApplicationContext
         * 构建方式:通过ClassPathXmlApplicationContext加载配置文件
         * 使用bean:getBean
         */
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean.xml");
        AccountDao accountDaoA = (AccountDao) applicationContext.getBean("accountDao");
        AccountDao accountDaoB = (AccountDao) applicationContext.getBean("accountDao");

        System.out.println("accountDaoA的内存地址:"+accountDaoA.hashCode());
        System.out.println("accountDaoB的内存地址:"+accountDaoB.hashCode());

    }

}


【3.2】Bean【デフォルト:シングルトン】

Bean タグを使用して、Bean.xml で accountDao のscope="singleton" をアセンブルします。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="accountDao" class="com.spring.dao.impl.AccountDaoImpl" scope="singleton"></bean>

    <!--创建accountServic-->
    <bean id="accountServic" class="com.spring.service.impl.AccountServicImpl"></bean>
</beans>
【3.3】シングルトン演算結果

ここに画像の説明を挿入します

【3.4】bean【多例:prototype】

Bean タグを使用して、bean.xml で accountDao のscope="prototype" をアセンブルします。

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
       xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
       xsi:schemaLocation="http://www.springframework.org/schema/beans
       http://www.springframework.org/schema/beans/spring-beans.xsd">
    <bean id="accountDao" class="com.spring.dao.impl.AccountDaoImpl" scope="prototype"></bean>

    <!--创建accountServic-->
    <bean id="accountServic" class="com.spring.service.impl.AccountServicImpl"></bean>
</beans>
【3.5】試作走行結果

ここに画像の説明を挿入します

【4】Beanスコープの概要
1、单例和多里创建方式、内存地址
	【singleton单例】:所有请求只创建一个对象,内存地址相同
	【prototype多例】:每次请求都创建新的对象,内存地址不同
2、为什么使用单例?
	节省内存、CPU的开销,加快对象访问速度
3、为什么使用多例?
	如果你给controller中定义很多的属性,那么单例肯定会出现竞争访问,不要在controller层中定义成员变量(dao、service注入的bean)
	当web层的对象是有状态的时候 使用多例,防止并发情况下的互相干扰
4、单例、多例的场景
	单例===》spring中的Dao,Service,controller都是单例的
	多例====》struts2的Action是多实例

4. Beanタグのライフサイクル

sevlet ライフサイクルのレビュー

1.被创建:执行init方法,只执行一次

  --默认情况下,第一次被访问时,Servlet被创建,然后执行init方法;

  --可以配置执行Servlet的创建时机;

2.提供服务:执行service的doGet、doPost方法,执行多次

3.被销毁:当Servlet服务器正常关闭时,执行destroy方法,只执行一次

Spring-IOC のさまざまなスコープの Bean のライフサイクル

範囲 ライフサイクル
シングルトンスコープ = "シングルトン" オブジェクトはすべてのリクエストに対して 1 回だけ作成されます Birth: アプリケーションがロードされ、コンテナーが作成され、オブジェクトが作成されます
。 Alive: コンテナーが存在する限り、オブジェクトは生き続けます。
死: アプリケーションはアンインストールされ、コンテナーは破棄され、オブジェクトは破棄されます。
多例scope=“prototype” リクエストごとにオブジェクトを作成する Birth: アプリケーションがロードされ、コンテナーが作成され、それを使用してオブジェクトが作成されます
Alive: コンテナーが存在する限り、オブジェクトは常に生きています。
死: オブジェクトは長期間使用されていないため、ガベージ コレクターによって収集されます。

ライフサイクルメソッド関連

名前 説明する
初期化メソッド クラス内の初期化メソッド名を指定します。
破壊メソッド クラス内で破棄メソッド名を指定する
【1】目標
1、掌握bean的生命周期配置方式
2、单例和多例下bean的生命周期的区别。
【2】Beanライフサイクルインスタンス
步骤:
	1、创建LifecycBeanServic类
	2、装配LifecycBeanServic
	3、创建测试类
	4、观察默认单例下生命周期
	5、观察多例下生命周期
【2.1】プロジェクトの作成

/**
 * @Description:生命周期测试服务
 */
public class LifecycleService {
    
    

    public LifecycleService() {
    
    
        System.out.println("LifecycleService构造");
    }

    public void init(){
    
    
        System.out.println("LifecycleService初始化");
    }

    public void doJob(){
    
    
        System.out.println("LifecycleService工作中");
    }

    public void destroy(){
    
    
        System.out.println("LifecycleService销毁");
    }
}

【2.3】LifecycleBeanの組み立て

Bean を接続してシングルトンをセットアップします。

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

    <bean id="lifecycBeanServic" class="com.spring.service.LifecycBeanServic"
          scope="singleton" init-method="init" destroy-method="destory"></bean>
</beans>
【2.4】ClientControllerの作成

/**
 * @Description:客户端
 */
public class ClientContrller {
    
    

    /**
     * ApplicationContext:spring-IOC容器
     * ClassPathXmlApplicationContext:容器实现类,加载配置文件
     * applicationContext.getBean:获得容器中的bean对象
     */
    @Test
    public void createAccount(){
    
    
        ApplicationContext applicationContext = new ClassPathXmlApplicationContext("bean.xml");
        System.out.println("applicationContext初始化完成");
        LifecycleService lifecycleService = applicationContext.getBean("lifecycleService", LifecycleService.class);
        lifecycleService.doJob();
        System.out.println("applicationContext容器关闭");
        ((ClassPathXmlApplicationContext) applicationContext).close();

    }
}

【2.5】シングルトンモードのライフサイクル:
【2.6】マルチインスタンスモードのライフサイクル

構成ファイル内の単一インスタンスを複数のインスタンスに変更します。

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

    <bean id="lifecycBeanServic" class="com.spring.service.LifecycBeanServic"
          scope="prototype"  init-method="init" destroy-method="destory"></bean>
</beans>

再度メソッドを実行すると、LifecycleBean が遅延ロードされており、初期化メソッド -init のみが実行され、破棄メソッド destroy が実行されていないことがわかりました。

【3】Beanのライフサイクルの概要
单例对象:scope="singleton"
	一个应用只有一个对象的实例。它的作用范围就是整个应用。
	生命周期:
		对象出生:当应用加载,创建容器时,对象就被创建了。
		对象活着:只要容器在,对象一直活着。
		对象死亡:当应用卸载,销毁容器时,对象就被销毁了。

多例对象:scope="prototype"
	每次访问对象时,都会重新创建对象实例。
	生命周期:
		对象出生:当使用对象时,创建新的对象实例(getBean)。
		对象活着:只要对象在使用中,就一直活着。
		对象死亡:当对象长时间不用时,被垃圾回收器回收。
生命周期方法:
init-method:指定类中的初始化方法名称 
destroy-method:指定类中销毁方法名称

5.Beanのインスタンス化方法

Bean をインスタンス化するには 3 つの方法があります。

  • Bean のデフォルト コンストラクターの作成
  • 静的ファクトリーメソッドの作成
  • ファクトリメソッドの作成をインスタンス化する

デフォルトのコンストラクタモード【ポイント】

設定方法
    <!--bean
        作用:
            声明类交给spring容器
        属性:
            id: 唯一标识
            class:全路径限定名称
        细节:
            默认使用无参构造函数实例化-->
    <bean id="connectionUtils" class="com.lagou.edu.utils.ConnectionUtils"></bean>
予防
缺省构造函数实例化Bean的方式是Spring中默认的实例化方式;
被实例化的Bean中必须有无参构造;
静的ファクトリーメソッドのアプローチ
設定方法
  <!--另外两种方式是为了我们自己new的对象加入到SpringIOC容器管理-->
    <!--方式二:静态方法-->
    <bean id="connectionUtils" class="com.lagou.edu.factory.CreateBeanFactory" factory-method="getInstanceStatic"/>
静的ファクトリーを作成する
public class CreateBeanFactory {
    
    
    public static ConnectionUtils getInstanceStatic() {
    
    
        return new ConnectionUtils();
    }
}
インスタンスファクトリーメソッド
設定方法
<!--实例化工厂实例化-->
<bean id="instanceFactory" class="com.spring.factory.InstanceFactory"></bean>
<bean id="accountInstance" factory-bean="instanceFactory" factory-method="createAccount"></bean>
インスタンスファクトリーの作成
/**
 * @Description:实例化工厂
 */
public class InstanceFactory {
    
    

    public Account createAccount(){
    
    
        System.out.println("实例工厂构建!");
        return  new Account();
    }
    
    public User createUser(){
    
    
        System.out.println("实例工厂构建!");
        return  new User();
    }
}

Bean のインスタンス化の概要
【缺省构造函数方式】
	说明:
		在默认情况下会根据默认缺省构造函数来创建类对象。如果bean中没有默认无参构造函数,将会创建失败。
	场景:
		当各个bean的业务逻辑相互比较独立时,或者与外界关联较少时可以使用

【静态工厂方法方式】
	说明:
        使用工厂中的静态方法创建对象,并装配到 spring的IOC 容器中。
        id 属性:指定 bean 的 id,用于从容器中获取   
        class 属性:指定静态工厂的全限定类名   
        factory-method 属性:指定生产对象的静态方法
	场景:
		统一管理各个bean的创建
		各个bean在创建之前需要相同的初始化处理,则可用静态factory方法进行统一的处理

【实例工厂方法方式】
	说明
        使用工厂中的实例方法创建对象,并装配到容器中。
            1、先把实例工厂做为一个bean装配到 spring容器中。   
            2、然后再引用工厂bean 来调用里面的非静态方法来获取bean并装配到spring的IOC容器中。   
            factory-bean 属性:用于指定实例工厂 bean 的 id。   
            factory-method 属性:用于指定实例工厂中创建对象的方法
	场景:
		1.实例factory方法也作为业务bean控制,可以用于集成其他框架的bean创建管理方法,
		2.能够使bean和factory的角色互换

セクション 2 Spring IOC の高度な機能

Spring IOC ソースコードの詳細な分析

おすすめ

転載: blog.csdn.net/weixin_43811057/article/details/129649199