Spring基於XML的IOC學習總結

IOC的環境配置

導入依賴

我們需要在我們的spring工程中pom.xml文件里加入IOC的相關依賴,依賴如下:

  <dependencies>
        <dependency>
            <groupId>org.springframework</groupId>
            <artifactId>spring-context</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
    </dependencies>

然後接下來的代碼示例,項目的代碼結構如下圖所示:
在這裏插入圖片描述

接下來,由於我們採用的是XML形式配置IOC,所以我們還需要一個XML文件來配置我們的beans。
resources目錄下,我們創建了一個名爲bean.xml文件。
bean.xml文件內容如下:

<?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">

    <!--把對象的創建交給spring來管理-->
    <bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"></bean>

    <bean id="accountDao" class="com.itheima.dao.impl.AccountDaoImpl"></bean>
</beans>

首先通過beans的標籤裏,導入有關ioc的相關約束。
然後在beans標籤內部,建立bean標籤,來配置我們的bean對象。
其中id爲我們給bean對象起的一個唯一標識名,class對於我們bean對象所對應的類。
在本文的代碼示例中,AccountServiceImpl爲業務層service接口實現類,AccountDaoImpl爲持久層dao實現類。

到此爲止,我們的有關IOC的相關配置已經完成。

IOC的使用

先看一下本文例子有關類的代碼
IAccountDao.java:

/**
 * 賬戶的持久層接口
 */
public interface IAccountDao {

    /**
     * 模擬保存賬戶
     */
    void saveAccount();
}

實現類AccountDaoImpl.java:

/**
 * 賬戶的持久層實現類
 */
public class AccountDaoImpl implements IAccountDao {

    public  void saveAccount(){

        System.out.println("保存了賬戶");
    }
}

IAccountService.java:

/**
 * 賬戶業務層的接口
 */
public interface IAccountService {

    /**
     * 模擬保存賬戶
     */
    void saveAccount();
}

AccountServiceImpl.java:

/**
 * 賬戶的業務層實現類
 */
public class AccountServiceImpl implements IAccountService {

    private IAccountDao accountDao ;

    public AccountServiceImpl(){
        System.out.println("對象創建了");
    }

    public void  saveAccount(){
        accountDao.saveAccount();
    }
}

最後,我們定義一個client類來模擬表現層,調用service:

**
 * 模擬一個表現層,用於調用業務層
 */
public class Client {

    public static void main(String[] args) {
        //1.獲取核心容器對象
        ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
       // ApplicationContext ac = new FileSystemXmlApplicationContext("C:\\Users\\zhy\\Desktop\\bean.xml");
        //2.根據id獲取Bean對象
        IAccountService as  = (IAccountService)ac.getBean("accountService");  //第一種獲取bean的方式
        IAccountDao adao = ac.getBean("accountDao",IAccountDao.class);  //第二種獲取bean的方式,兩者等價

        as.saveAccount();

    }
}

ApplicationContext接口爲spring提供的有關IOC的接口,還有一個名爲BeanFactory的接口,這兩個接口的區別爲:

ApplicationContext: 單例對象適用 採用此接口
它在構建核心容器時,創建對象採取的策略是採用立即加載的方式。也就是說,只要一讀取完配置文件馬上就創建配置文件中配置的對象。

BeanFactory: 多例對象使用
它在構建核心容器時,創建對象採取的策略是採用延遲加載的方式。也就是說,什麼時候根據id獲取對象了,什麼時候才真正的創建對象。

ApplicationContext的三個常用實現類:

  • ClassPathXmlApplicationContext:它可以加載類路徑下的配置文件,要求配置文件必須在類路徑下。不在的話,加載不了。(更常用)
  • FileSystemXmlApplicationContext:它可以加載磁盤任意路徑下的配置文件(必須有訪問權限)
  • AnnotationConfigApplicationContext:它是用於讀取註解創建容器的

創建bean的三種方式

第一種方式:使用默認構造函數創建
也就是我們上面的例子所用到的,例如

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

在spring的配置文件中使用bean標籤,配以id和class屬性之後,且沒有其他屬性和標籤時。
採用的就是默認構造函數創建bean對象,此時如果類中沒有默認構造函數,則對象無法創建。

第二種方式: 使用普通工廠中的方法創建對象(使用某個類中的方法創建對象,並存入spring容器)

我們可以模擬一個工廠類,該類可能是存在於jar包中的,我們無法通過修改源碼的方式來提供默認構造函數。此時,我們需要調用該類中的一個方法,才能得到我們想用到的類,例如:

public class InstanceFactory {

    public IAccountService getAccountService(){
        return new AccountServiceImpl();
    }
}

我們需要通過調用InstanceFactory類中的getAccountService函數得到我們想要的對象,此時,我們就用到了第二種方式,例:

<bean id="instanceFactory" class="com.itheima.factory.InstanceFactory"></bean>
    <bean id="accountService" factory-bean="instanceFactory" factory-method="getAccountService"></bean>

通過第一個bean標籤,將InstanceFactory導入。
然後第二個bean標籤,factory-bean屬性指定調用的類的bean標籤id,factory-method對應調用的方法。

第三種方式:使用工廠中的靜態方法創建對象(使用某個類中的靜態方法創建對象,並存入spring容器)
與第二種方式大同小異,直接看代碼示例:

<bean id="accountService" class="com.itheima.factory.StaticFactory" factory-method="getAccountService"></bean>

因爲是要調用一個類的靜態方法,所以我們不需要先將該類導入爲bean對象,直接在class標籤中指定該類就可以。

bean的作用範圍調整

bean標籤的scope屬性:

作用:用於指定bean的作用範圍
取值: 常用的就是單例的和多例的
        singleton:單例的(默認值)
        prototype:多例的
        request:作用於web應用的請求範圍
        session:作用於web應用的會話範圍
        global-session:作用於集羣環境的會話範圍(全局會話範圍),當不是集羣環境時,它就是session

例如:

 <bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl" scope="prototype"></bean>

bean對象的生命週期

單例對象
         出生:當容器創建時對象出生
         活着:只要容器還在,對象一直活着
         死亡:容器銷燬,對象消亡
         總結:單例對象的生命週期和容器相同
多例對象
         出生:當我們使用對象時spring框架爲我們創建
         活着:對象只要是在使用過程中就一直活着。
         死亡:當對象長時間不用,且沒有別的對象引用時,由Java的垃圾回收器回收

bean標籤的init-method屬性可以指定bean對象初始化時,執行某個函數,destroy-method屬性可以指定bean對象摧毀時,執行某個函數。例如:
AccountServiceImpl.java:

public class AccountServiceImpl implements IAccountService {



    public AccountServiceImpl(){
        System.out.println("對象創建了");
    }

    public void  saveAccount(){
        System.out.println("service中的saveAccount方法執行了。。。");
    }

    public void  init(){
        System.out.println("對象初始化了。。。");
    }
    public void  destroy(){
        System.out.println("對象銷燬了。。。");
    }

}

對應的bean標籤:

 <bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl"
          scope="prototype" init-method="init" destroy-method="destroy"></bean>

spring中的依賴注入

依賴注入:Dependency Injection
IOC的作用:降低程序間的耦合(依賴關係)
依賴關係的管理:以後都交給spring來維護
在當前類需要用到其他類的對象,由spring爲我們提供,我們只需要在配置文件中說明
依賴關係的維護,就稱之爲依賴注入。
依賴注入:
能注入的數據:有三類

  • 基本類型和String
  • 其他bean類型(在配置文件中或者註解配置過的bean)
  • 複雜類型/集合類型

注入的方式:有三種

  • 第一種:使用構造函數提供
  • 第二種:使用set方法提供
  • 第三種:使用註解提供(xml方式中先不介紹)

第一種,使用構造函數提供:

使用的標籤:**constructor-arg**
標籤出現的位置:bean標籤的內部
標籤中的屬性
      type:用於指定要注入的數據的數據類型,該數據類型也是構造函數中某個或某些參數的類型
      index:用於指定要注入的數據給構造函數中指定索引位置的參數賦值。索引的位置是從0開始
      name:用於指定給構造函數中指定名稱的參數賦值     
  =============以上三個用於指定給構造函數中哪個參數賦值===============
     value:用於提供基本類型和String類型的數據
     ref:用於指定其他的bean類型數據。它指的就是在spring的Ioc核心容器中出現過的bean對象

優勢:
在獲取bean對象時,注入數據是必須的操作,否則對象無法創建成功。
弊端:
改變了bean對象的實例化方式,使我們在創建對象時,如果用不到這些數據,也必須提供。

例如:
AccountServiceImpl.java:

public class AccountServiceImpl implements IAccountService {

    //如果是經常變化的數據,並不適用於注入的方式
    private String name;
    private Integer age;
    private Date birthday;

    public AccountServiceImpl(String name,Integer age,Date birthday){
        this.name = name;
        this.age = age;
        this.birthday = birthday;
    }

    public void  saveAccount(){
        System.out.println("service中的saveAccount方法執行了。。。"+name+","+age+","+birthday);
    }


}

對應bean標籤:

<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl">
  <constructor-arg name="name" value="泰斯特"></constructor-arg>
  <constructor-arg name="age" value="18"></constructor-arg>
  <constructor-arg name="birthday" ref="now"></constructor-arg>
</bean>

第二種,set方法注入:
使用set方法注入時,我們需要在類中,爲所有需要被注入的對象,設置set方法,例如:

public class AccountServiceImpl2 implements IAccountService {

    //如果是經常變化的數據,並不適用於注入的方式
    private String name;
    private Integer age;
    private Date birthday;

    public void setName(String name) {
        this.name = name;
    }

    public void setAge(Integer age) {
        this.age = age;
    }

    public void setBirthday(Date birthday) {
        this.birthday = birthday;
    }

    public void  saveAccount(){
        System.out.println("service中的saveAccount方法執行了。。。"+name+","+age+","+birthday);
    }


}

set方法注入                更常用的方式
   涉及的標籤:property
   出現的位置:bean標籤的內部
   標籤的屬性
      name:用於指定注入時所調用的set方法名稱
      value:用於提供基本類型和String類型的數據
      ref:用於指定其他的bean類型數據。它指的就是在spring的Ioc核心容器中出現過的bean對象
   優勢:
       創建對象時沒有明確的限制,可以直接使用默認構造函數
   弊端:
       如果有某個成員必須有值,則獲取對象是有可能set方法沒有執行。
<bean id="accountService2" class="com.itheima.service.impl.AccountServiceImpl2">
        <property name="name" value="TEST" ></property>
        <property name="age" value="21"></property>
        <property name="birthday" ref="now"></property>
    </bean>

複雜類型的注入:
用於給List結構集合注入的標籤:
list array set
用於個Map結構集合注入的標籤:
map props

例如:
AccountServiceImpl3.java:

public class AccountServiceImpl3 implements IAccountService {

    private String[] myStrs;
    private List<String> myList;
    private Set<String> mySet;
    private Map<String,String> myMap;
    private Properties myProps;

    public void setMyStrs(String[] myStrs) {
        this.myStrs = myStrs;
    }

    public void setMyList(List<String> myList) {
        this.myList = myList;
    }

    public void setMySet(Set<String> mySet) {
        this.mySet = mySet;
    }

    public void setMyMap(Map<String, String> myMap) {
        this.myMap = myMap;
    }

    public void setMyProps(Properties myProps) {
        this.myProps = myProps;
    }

    public void  saveAccount(){
        System.out.println(Arrays.toString(myStrs));
        System.out.println(myList);
        System.out.println(mySet);
        System.out.println(myMap);
        System.out.println(myProps);
    }


}

對應bean代碼:

<bean id="accountService3" class="com.itheima.service.impl.AccountServiceImpl3">
        <property name="myStrs">
            <set>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </set>
        </property>

        <property name="myList">
            <array>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </array>
        </property>

        <property name="mySet">
            <list>
                <value>AAA</value>
                <value>BBB</value>
                <value>CCC</value>
            </list>
        </property>

        <property name="myMap">
            <props>
                <prop key="testC">ccc</prop>
                <prop key="testD">ddd</prop>
            </props>
        </property>

        <property name="myProps">
            <map>
                <entry key="testA" value="aaa"></entry>
                <entry key="testB">
                    <value>BBB</value>
                </entry>
            </map>
        </property>
    </bean>
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章