【Spring Boot】第5課-元件的控制反轉、依賴注入與抽換

本文最後更新於:2025-03-22

控制反轉(IoC)與依賴注入(DI)是 Spring Boot 中的重要觀念。而筆者選擇在第 4 課(三層式架構)結束,練習用專案的架構成形後,才開始介紹。

本文首先透過範例專案,讓讀者知道裡頭那些用來封裝程式邏輯的物件,其實存在著依賴關係。接著說明在後端程式運行期間,為何要求這些物件只能存在唯一一個,以及如何做到。

有了先備的概念後,筆者再經由這些議題,開始介紹控制反轉與依賴注入。最後引進物件導向的「多型」特性,示範以介面來操作物件,有助於實作細節的抽換。


本文的練習用專案,請點我

一、依賴關係

在介紹控制反轉與依賴注入的概念前,讓我們先看看範例專案大概的架構。

以下是產品和使用者的 Repository 層,它們使用 Java 的 Map 資料結構代替真實 DB。並且提供一些方法供外部呼叫,藉此對資料做存取。

public class ProductRepository {
    private static final Map<String, ProductPO> productMap = new HashMap<>();

    public ProductPO getOneById(String id) {
        return productMap.get(id);
    }

    // ...
}
public class UserRepository {
    private static final Map<String, UserPO> userMap = new HashMap<>();

    public UserPO getOneById(String id) {
        return userMap.get(id);
    }

    // ...
}

以下是產品的 Service 層,提供了商業邏輯。該類別擁有全域的 Repository 物件,讓商業邏輯的程式碼呼叫。

public class ProductService {
    private static final ProductRepository productRepository = new ProductRepository();
    private static final UserRepository userRepository = new UserRepository();

    public ProductVO getProductVO(String id) {
        var productPO = productRepository.getOneById(id);
        // ...

        var userPO = userRepository.getOneById(productPO.getCreatorId());
        // ...

        return productVO;
    }

    // ...
}

以下是 Controller 層,提供了 RESTful API。該類別擁有全域的 Service 物件。

@RestController
@RequestMapping(path = "/products")
public class ProductController {
    private static final ProductService productService = new ProductService();
    // ...
}

讀者可以看出它們之間的關係:Controller 呼叫 Service;而 Service 呼叫 Repository。這種「誰呼叫誰」的關係,正式的稱呼為「依賴」(depend)。

附帶一提,範例專案為了簡便,並未實作使用者的 Controller 和 Service。不然原則上也會是「UserController」依賴「UserService」;而 UserService 依賴 UserRepository 的關係。

@RestController
public class UserController {
    private static final UserService userService = new UserService();
    // ...
}
public class UserService {
    private static final UserRepository userRepository = new UserRepository();
    // ...
}

二、元件與單例的概念

本節將討論,同樣是封裝程式邏輯,為何要特別建立物件呢?並且向讀者介紹「單例」(singleton)的概念。

(一)元件

無論是 ProductService、ProductRepository 或 UserRepository,雖然它們都被建立成物件,但目的都是封裝程式邏輯,而非攜帶資料到處傳遞。

像這種用途的物件,我們給它一個更正式的稱呼,叫做「元件」,英文為「bean」或「component」。元件可以提供方法,來實現商業邏輯、資料處理或存取 DB 等功能。

既然只是用來封裝邏輯,那為什麼不宣告成靜態(static)的方法就好呢?這樣連物件都不必建立了。示意如下:

public class ProductService {
    private ProductService() {}

    // 宣告為靜態方法
    public static ProductVO getProductVO(String id) {
        // ...
    }
}
@RestController
@RequestMapping(path = "/products")
public class ProductController {
    
    @GetMapping("/{id}")
    public ResponseEntity<ProductVO> getProduct(@PathVariable("id") String productId) {
        // 呼叫靜態方法
        var product = ProductService.getProductVO(productId);

        // ...
    }
}

要知道,系統功能是可以很複雜的,若元件一律提供靜態方法,那就不能善用物件導向中,繼承與多型的特性了。

(二)單例

筆者先前建立元件的全域變數,其用意是避免每呼叫一次方法,就建立一次元件,示意如下:

public class ProductService {

    public ProductVO getProductVO(String id) {
        var productPO = new ProductRepository().getOneById(id);
        // ...

        var userPO = new UserRepository().getOneById(productPO.getCreatorId());
        // ...
    }

    // ...
}

建立物件會在記憶體佔一個空間,而用完就又要回收,其實沒必要如此反覆。更別提在用戶多的系統,伺服器是很忙碌的。

為了確保應用程式運行期間,特定類別的物件只會存在一個,並能讓各個地方共享,於是出現了「單例」的概念。「單」是單一的意思,而「例」是實例(instance)。

讀者在網路上,能找到各種單例的程式寫法,以下是簡單的範例:

public class ProductRepository {
    private static final ProductRepository INSTANCE  = new ProductRepository();

    private ProductRepository() {
        // 初始化...
    }

    // 提供取得唯一物件的方法
    public static ProductRepository getInstance() {
        return INSTANCE;
    }
}

但也能顧及執行緒安全而變得複雜:

public class ProductRepository {
    private static ProductRepository INSTANCE;

    private ProductRepository() {
        // 初始化...
    }

    // 提供取得唯一物件的方法
    public static ProductRepository getInstance() {
        if (INSTANCE == null) {
            synchronized (ProductRepository.class) {
                if (INSTANCE == null) {
                    INSTANCE = new ProductRepository();
                }
            }
        }
        return INSTANCE;
    }
}

元件的依賴關係,可能會無意間建立出多餘的物件。而手動實現單例,又很不方便。幸好 Spring Boot 有提供「控制反轉」和「依賴注入」的功能,來解決這些問題。

三、控制反轉(Inversion of Control,IoC)

在 Java 語言中建立物件的方式,是在想要的地方使用 new 關鍵字。而「控制反轉」的精神,則是將建立物件的工作轉移給外界。

也就是說,無論是透過建構子或 setter 方法,只要物件並非在類別內部建立,而是由外部提供,那就可以稱之為控制反轉。

那要如何讓 Spring Boot 建立單例物件呢?做法是在類別冠上特定的注解(annotation)。Spring Boot 啟動時,會透過 Java 的「反射」(reflection)機制,掃描專案中的哪些類別具有這些注解。

可使用的注解如下,它們有不同的涵義。

  • @Controller:代表提供 Web API 的表現層。@RestController 注解便是繼承於它。於第 3.1 課介紹。
  • @Service:代表商業邏輯層。
  • @Repository:代表資料存取層。
  • @Configuration:代表這裡搭配 @Value 注解,存放了「application.properties」配置檔的值。於第 6 課介紹。或者搭配 @Bean 注解,控制元件的初始化過程。於第 7 課介紹。
  • @Component:以上 4 項皆繼承自此注解,是泛用的選擇。

接著,我們在範例專案中的元件類別,冠上適當的注解。

@Service
public class ProductService {
    // ...
}
@Repository
public class ProductRepository {
    // ...
}
@Repository
public class UserRepository {
    // ...
}

如此一來,Spring Boot 建立出單例物件後,便會放在記憶體中管理。這個地方稱為「IoC 容器」。為了方便說明,筆者後續都用「元件」來指稱 IoC 容器中的單例物件。

四、依賴注入(Dependency Injection,DI)

(一)介紹

第一節展示的範例專案,說明了元件的依賴關係。而依賴注入便是要取代原先在程式碼中 new 物件的做法。

類別必須具備前面提到的注解,才能被建立為元件。而 Spring Boot 可以把這些元件「注入」到其他的元件。

範例程式中的 ProductService 依賴 ProductRepository 與 UserRepository。因此這些 Repository 元件,就會被注入到 ProductService 元件中。而 ProductController 又依賴 ProductService,故 Controller 也會被注入。

我們可以使用 @Autowired 注解,讓 Spring Boot 注入元件。根據使用此注解的方式,又分為「欄位注入」與「建構子注入」。

(二)欄位注入(Field Injection)

欄位注入的做法,是對元件的全域變數冠上 @Autowired 注解。

@RestController
@RequestMapping(path = "/products")
public class ProductController {

    @Autowired
    private ProductService productService;
    
    // ...
}
@Service
public class ProductService {

    @Autowired
    private ProductRepository productRepository;

    @Autowired
    private UserRepository userRepository;

    // ...
}

Spring Boot 會先建立出所有的元件,接著才把每個元件所依賴的其他元件逐一注入。因此元件可能有短暫時間處於初始化不完整的空窗期,這是一項缺點。而優點是寫法方便。

(三)建構子注入(Constructor Injection)

建構子注入的做法,是宣告包含所有依賴的建構子,再對其加上 @Autowired 注解。

@Service
public class ProductService {
    private final ProductRepository productRepository;
    private final UserRepository userRepository;

    @Autowired
    public ProductService(ProductRepository productRepository, UserRepository userRepository) {
        this.productRepository = productRepository;
        this.userRepository = userRepository;
    }
    // ...
}

Spring Boot 在建立一個元件時,會先確認它所依賴的其他元件是否都建立好了。是的話,便從建構子注入進來。否則就先建立其他元件。意即建立與注入元件是同時進行的。

這項做法的缺點是讓程式碼變得較冗長。然而瑕不掩瑜,優點是確保元件在被使用時,已經處於完整初始化的狀態。另外也有利於撰寫單元測試(unit test),因為我們可以將設計好的「模擬物件」(mock)由建構子傳入。

Spring 官方也建議採取這樣的注入方式。

五、使用介面注入元件

為了善用物件導向的「多型」特性,我們可以讓元件的類別實作「介面」。而進行依賴注入時,則以介面代替類別。這樣有助於未來更換元件時,不會影響到外部使用的方式。

(一)多型呼叫

請讀者透過 Java 語言回想一下,當我們讓類別實作「介面」(interface)時,必須完成介面所定義的 public 方法。

當建立這種類別的物件時,宣告的型態可以用介面來取代類別。比方說「ArrayList」與「LinkedList」這兩種資料結構,均實作「List」介面。

用 List 宣告後,使用物件一律都是呼叫該介面所定義的方法。而不必在意實際上是 ArrayList 或者 LinkedList 在運作。示意如下:

public static void main(String[] args) {
    List<String> arrayList = new ArrayList<>();
    List<String> linkedList = new LinkedList<>();
    process(arrayList);
    process(linkedList);
}

private static void process(List<String> list) {
    list.add("A");
    list.add("B");
    list.add("C");

    for (var i = 0; i < list.size(); i++) {
        System.out.println(list.get(i));
    }

    list.clear();
}

回到範例專案,ProductRepository 使用 Map 資料結構來儲存資料。我們同樣也能為 Repository 層設計一個介面,並開發出其他儲存方式的元件。例如用 List 儲存,甚至連接到真實的 DB。

而 ProductService 則固定以該介面操作 Repository 元件。

(二)實作介面注入

以下設計一個叫做「IProductRepository」的介面,並進行實作。此處為避免混淆,故將原先的 ProductRepository 改名為「MapProductRepository」,強調儲存方式。

public interface IProductRepository {
    ProductPO getOneById(String id);
    ProductPO insert(ProductPO product);
    void update(ProductPO product);
    void deleteById(String id);
    List<ProductPO> getMany(ProductRequestParameter param);
}
@Repository
public class MapProductRepository implements IProductRepository {
    private static final Map<String, ProductPO> productMap = new HashMap<>();
    // ...

    public ProductPO getOneById(String id) {
        return productMap.get(id);
    }

    public void deleteById(String id) {
        productMap.remove(id);
    }

    // ...
}

接著調整 ProductService,改以 IProductRepository 介面來注入。

@Service
public class ProductService {
    private final IProductRepository productRepository;
    private final UserRepository userRepository;

    public ProductService(IProductRepository productRepository, UserRepository userRepository) {
        this.productRepository = productRepository;
        this.userRepository = userRepository;
    }

    // ...
}

經過這樣的修改,ProductService 呼叫的都是 IProductRepository 介面的方法,而實際執行的是被注入元件的內部邏輯。

現在 Spring Boot 啟動時,就會尋找專案中有哪些元件實作了 IProductRepository 介面。目前只有一個,理所當然注入 MapProductRepository。

六、抽換相同介面的元件

為了說明如何在相同介面更換元件,筆者準備了另一個實作 IProductRepository 介面的元件。下面是以 List 來儲存產品資料,程式碼寫法將完全不同。

@Repository
public class ListProductRepository implements IProductRepository {
    private static final List<ProductPO> productList = new ArrayList<>();
    // ...
    
    public ProductPO getOneById(String id) {
        return productList
                .stream()
                .filter(p -> p.getId().equals(id))
                .findFirst()
                .orElse(null);
    }

    public void deleteById(String id) {
        productList.removeIf(p -> p.getId().equals(id));
    }

    // ...
}

上面的 ListProductRepository 需依照介面的規範,完成所有 public 方法。完整的範例程式,請參考文末附上的專案。

此時專案中存在多個相同介面的元件。由於 Spring Boot 不知道要注入哪一個,所以啟動會失敗,錯誤訊息節錄如下:

Parameter 0 of constructor in com.example.demo.service.ProductService required a single bean, but 2 were found:
    - listProductRepository: ...
    - mapProductRepository: ...
...
Consider marking one of the beans as @Primary, updating the consumer to accept multiple beans, or using @Qualifier to identify the bean that should be consumed

為了告訴 Spring Boot 要注入哪一個元件,我們可在其中一個類別冠上 @Primary 注解。

@Repository
@Primary
public class ListProductRepository implements IProductRepository {
    // ...
}

這會讓所有依賴 IProductRepository 介面的地方,都注入 ListProductRepository。

或者也能透過 @Qualifier 注解,在不同的元件注入不同的依賴。

@Service
public class ProductService {
    private final IProductRepository productRepository;
    private final UserRepository userRepository;

    @Autowired
    public ProductService(@Qualifier("mapProductRepository") IProductRepository productRepository,
                          UserRepository userRepository) {
        this.productRepository = productRepository;
        this.userRepository = userRepository;
    }

    // ...
}

以上是在建構子注入的場合使用 @Qualifier 注解,需傳入「元件名稱」作為參數。

元件名稱預設是類別名稱的駝峰字。若想自定義,可在 @Repository@Service 等元件的注解,傳入參數做設定。

@Repository("mapProductRepository")
public class MapProductRepository implements IProductRepository {
    // ...
}

至於在欄位注入的場合使用 @Qualifier 注解,則與 @Autowired 注解一併冠在全域變數之上即可。

@Service
public class ProductService {

    @Autowired
    @Qualifier("mapProductRepository")
    private final IProductRepository productRepository;

    // ...
}

本文的完成後專案,請點我

上一課:【Spring Boot】第4課-實作三層式架構的 Service 與 Repository

下一課:【Spring Boot】第6課-在 application.properties 配置檔提供設定值(以 Java Mail 為例)


張貼留言: