按照我們總結的TDD的開發流程,開始進行后臺開發。由于我們當前為前后臺分離的架構,所以省略了原型開發,而直接由第二步功能代碼初始化做起。
由前臺的開發稍有不同的是,在后臺的開發中我們已經進行了更合理的分層,在分層開發時我們需要依據每個層具體負責的功能來編寫針對性的測試語句。
## 初始化
controller/KlassController.java
```
@DeleteMapping("{id}")
@ResponseStatus(HttpStatus.NO_CONTENT)
public void delete(@PathVariable Long id) {
klassService.deleteById(id);
}
```
service/KlassService.java
```
/**
* 刪除
*
* @param id 班級ID
*/
void deleteById(Long id);
```
service/KlassServiceImpl.java
```
@Override
public void deleteById(Long id) {
}
```
## M層
在初始化中,我們分別建立了KlassController以及KlassServiceImpl。而KlassController依賴于KlassServiceImpl,所以在開發中我們一般會優先開發更底層的KlassServiceImpl。
### 單元測試
同第一次測試KlassController的方法一樣,我們找到 KlassServiceImpl然后借助于idea來快速的生成測試文件KlassServiceImplTest.java

我們繼續添加delete測試方法后如下:
```
package com.mengyunzhi.springBootStudy.service;
import org.junit.Test;
public class KlassServiceImplTest {
@Test
public void deleteById() {
}
}
```
初始化:
```
package com.mengyunzhi.springBootStudy.service;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
@SpringBootTest
@RunWith(SpringRunner.class)
public class KlassServiceImplTest {
@Autowired KlassService klassService;
/**
* 1. 新建一個班級
* 2. 刪除這個班級
* 3. 再次查詢該班級
* 4. 斷言該班級不存在
*/
@Test
public void deleteById() {
}
}
```
依據注釋我們開始編寫測試功能,由于數據的增改查刪需要數據倉庫的協助,所以我們在此需要自動裝入KlassRepository:
```
import org.assertj.core.api.Assertions; ?
@Autowired
KlassRepository klassRepository;
/**
* 1. 新建一個班級
* 2. 刪除這個班級
* 3. 再次查詢該班級
* 4. 斷言該班級不存在
*/
@Test
public void deleteById() {
Klass klass = new Klass();
klass.setName("測試班級");
klassRepository.save(klass);
klassService.delete(klass.getId());
Optional<Klass>? klassOptional = klassRepository.findById(klass.getId());
Assertions?.assertThat(klassOptional.isPresent()?).isFalse(); ?
}
```
* ? AssertJ是一款優秀的單元測試工具,被廣泛的用于java的單元測試中。
* ? Optional是一個可以裝入其它對象的容器
* ? 該容器中的isPresent()用于判斷容器中當前是否裝入了對象。
* ? 斷言:此時容器中未裝入任何對象,即`klassRepository.findById(klass.getId())`未查詢出對應的班級。
啟動單元測試后結果如下:
```
org.junit.ComparisonFailure:
Expected :false
Actual :true
<Click to see difference>
...
at com.mengyunzhi.springBootStudy.service.KlassServiceImplTest.delete(KlassServiceImplTest.java:36)
```
提示我們在KlassServiceImplTest.java的36的斷言是失敗的。我們斷言是`false`,但實際的值是`true`。
### 功能代碼
使用JPA進行刪除操作時異常簡單:
KlassServiceImpl
```
@Override
public void deleteById(Long id) {
this.klassRepository.deleteById(id);?
}
```
* ? 直接調用deleteById()即可
#### 再測試

單元測試通過,說明功能符合預期。
## C層
前面我們描述過,C層的功能主要為:數據輸入、數據轉發以及數據輸出。而邏輯實現屬于M層的功能,所以C層并不需要對由M層實現的邏輯實現做斷言。
### 單元測試
以當前刪除為例,C層只要保證調用了M層的deleteById即為成功。而至于調用M層的deleteById后是否真正的發生了刪除操作、刪除操作是否成功,C層不負責也不關心。因此,在測試中我們也會像前臺一樣MOCK一個M層出來專門供測試使用。則按上述的原則,初始化如下:
KlassControllerTest
```
/**
* 刪除班級測試,測試點:
* 1. 特定URL的DELETE請求是否正常(輸入)
* 2. 是否返回了204狀態碼(輸出)
* 3. 是否成功的進行了數據轉發(數據轉發)
* @throws Exception
*/
@Test
public void delete() throws Exception {
}
```
接下來我們添加一個模擬KlassService的對象
```
import org.mockito.Mockito;
@MockBean ?
KlassService klassService;
```
* ? 注入一個實現了所有的KlassService接口規定功能的對象,并把該對象起名為klassService。
* ? 該對象的作用域為整個測試。此對象也將用于KlassController中注入KlassService。
使用 **@Autowired** 來裝配我們自己編寫的基于`KlassServiceImp`實現類創建的對象;使用 **@MockBean** 來裝入由Mockito自動根據接口來生成的對象。
*****
**接口**是一種規范、一種協議,現實生活中它是一種描述,用于對接兩種不同的產品、事物、物體、數據等。比如我們USB接口、3.5mm耳機接口,再比如因特網協議(IP),傳輸控制協議(TCP);而**類**則是藍圖、是模板,比如汽車生產圖紙、建筑圖紙等;而**對象**則指具體某些功能和屬性能的能動體,比如某輛汽車、比如某幢高樓、比如某個U盤。
正是由于我們在KlassController的聲明方法為:我需要一個有KlassService規定功能是對象,才會使得我們可以:① 在運行環境中,由Spring為其注入了基于KlassServiceImpl實例化的對象來滿足其要求;② 在測試環境中,由Mockito為求注入了一個實現了KlassService規定所有的功能的對象來滿足其要求。
> 我們把這種在編碼過程中指定**功能要求**而不是直接引用某個**對象**的方法稱為**面向接口**開發。
*****
### 完成測試代碼
```
@Test
public void delete() throws Exception {
String url = "/Klass/1";
MockHttpServletRequestBuilder delete = MockMvcRequestBuilders.delete(url); ?
this.mockMvc.perform(delete)
.andExpect(MockMvcResultMatchers.status().is(204)); ?
Mockito.verify(klassService).deleteById(1L); ?
}
```
* ? 構建一個delete請求,請求地址為/Klass/1 (輸入)
* ? 模擬發起請求,斷言返回的狀態碼為204(輸出)
* ? 斷言執行了klassService中的deleteById方法,且傳入的參數為1(數據轉發)
#### 測試

測試通過,說明我們在C層初始化時的代碼足夠的健壯。對輸入、輸出、數據轉發都做了正確的處理。
# 參考文檔
| 名稱 | 鏈接 | 預計學習時長(分) |
| --- | --- | --- |
| 源碼地址 | [https://github.com/mengyunzhi/spring-boot-and-angular-guild/releases/tag/step3.6.2](https://github.com/mengyunzhi/spring-boot-and-angular-guild/releases/tag/step3.6.2) | - |
| Mockito 專注于在測試中模擬服務 | [https://site.mockito.org/](https://site.mockito.org/) | - |
| AssertJ 專注于測試斷言 | [https://assertj.github.io/doc/](https://assertj.github.io/doc/) | - |
- 序言
- 第一章:Hello World
- 第一節:Angular準備工作
- 1 Node.js
- 2 npm
- 3 WebStorm
- 第二節:Hello Angular
- 第三節:Spring Boot準備工作
- 1 JDK
- 2 MAVEN
- 3 IDEA
- 第四節:Hello Spring Boot
- 1 Spring Initializr
- 2 Hello Spring Boot!
- 3 maven國內源配置
- 4 package與import
- 第五節:Hello Spring Boot + Angular
- 1 依賴注入【前】
- 2 HttpClient獲取數據【前】
- 3 數據綁定【前】
- 4 回調函數【選學】
- 第二章 教師管理
- 第一節 數據庫初始化
- 第二節 CRUD之R查數據
- 1 原型初始化【前】
- 2 連接數據庫【后】
- 3 使用JDBC讀取數據【后】
- 4 前后臺對接
- 5 ng-if【前】
- 6 日期管道【前】
- 第三節 CRUD之C增數據
- 1 新建組件并映射路由【前】
- 2 模板驅動表單【前】
- 3 httpClient post請求【前】
- 4 保存數據【后】
- 5 組件間調用【前】
- 第四節 CRUD之U改數據
- 1 路由參數【前】
- 2 請求映射【后】
- 3 前后臺對接【前】
- 4 更新數據【前】
- 5 更新某個教師【后】
- 6 路由器鏈接【前】
- 7 觀察者模式【前】
- 第五節 CRUD之D刪數據
- 1 綁定到用戶輸入事件【前】
- 2 刪除某個教師【后】
- 第六節 代碼重構
- 1 文件夾化【前】
- 2 優化交互體驗【前】
- 3 相對與絕對地址【前】
- 第三章 班級管理
- 第一節 JPA初始化數據表
- 第二節 班級列表
- 1 新建模塊【前】
- 2 初識單元測試【前】
- 3 初始化原型【前】
- 4 面向對象【前】
- 5 測試HTTP請求【前】
- 6 測試INPUT【前】
- 7 測試BUTTON【前】
- 8 @RequestParam【后】
- 9 Repository【后】
- 10 前后臺對接【前】
- 第三節 新增班級
- 1 初始化【前】
- 2 響應式表單【前】
- 3 測試POST請求【前】
- 4 JPA插入數據【后】
- 5 單元測試【后】
- 6 惰性加載【前】
- 7 對接【前】
- 第四節 編輯班級
- 1 FormGroup【前】
- 2 x、[x]、{{x}}與(x)【前】
- 3 模擬路由服務【前】
- 4 測試間諜spy【前】
- 5 使用JPA更新數據【后】
- 6 分層開發【后】
- 7 前后臺對接
- 8 深入imports【前】
- 9 深入exports【前】
- 第五節 選擇教師組件
- 1 初始化【前】
- 2 動態數據綁定【前】
- 3 初識泛型
- 4 @Output()【前】
- 5 @Input()【前】
- 6 再識單元測試【前】
- 7 其它問題
- 第六節 刪除班級
- 1 TDD【前】
- 2 TDD【后】
- 3 前后臺對接
- 第四章 學生管理
- 第一節 引入Bootstrap【前】
- 第二節 NAV導航組件【前】
- 1 初始化
- 2 Bootstrap格式化
- 3 RouterLinkActive
- 第三節 footer組件【前】
- 第四節 歡迎界面【前】
- 第五節 新增學生
- 1 初始化【前】
- 2 選擇班級組件【前】
- 3 復用選擇組件【前】
- 4 完善功能【前】
- 5 MVC【前】
- 6 非NULL校驗【后】
- 7 唯一性校驗【后】
- 8 @PrePersist【后】
- 9 CM層開發【后】
- 10 集成測試
- 第六節 學生列表
- 1 分頁【后】
- 2 HashMap與LinkedHashMap
- 3 初識綜合查詢【后】
- 4 綜合查詢進階【后】
- 5 小試綜合查詢【后】
- 6 初始化【前】
- 7 M層【前】
- 8 單元測試與分頁【前】
- 9 單選與多選【前】
- 10 集成測試
- 第七節 編輯學生
- 1 初始化【前】
- 2 嵌套組件測試【前】
- 3 功能開發【前】
- 4 JsonPath【后】
- 5 spyOn【后】
- 6 集成測試
- 7 @Input 異步傳值【前】
- 8 值傳遞與引入傳遞
- 9 @PreUpdate【后】
- 10 表單驗證【前】
- 第八節 刪除學生
- 1 CSS選擇器【前】
- 2 confirm【前】
- 3 功能開發與測試【后】
- 4 集成測試
- 5 定制提示框【前】
- 6 引入圖標庫【前】
- 第九節 集成測試
- 第五章 登錄與注銷
- 第一節:普通登錄
- 1 原型【前】
- 2 功能設計【前】
- 3 功能設計【后】
- 4 應用登錄組件【前】
- 5 注銷【前】
- 6 保留登錄狀態【前】
- 第二節:你是誰
- 1 過濾器【后】
- 2 令牌機制【后】
- 3 裝飾器模式【后】
- 4 攔截器【前】
- 5 RxJS操作符【前】
- 6 用戶登錄與注銷【后】
- 7 個人中心【前】
- 8 攔截器【后】
- 9 集成測試
- 10 單例模式
- 第六章 課程管理
- 第一節 新增課程
- 1 初始化【前】
- 2 嵌套組件測試【前】
- 3 async管道【前】
- 4 優雅的測試【前】
- 5 功能開發【前】
- 6 實體監聽器【后】
- 7 @ManyToMany【后】
- 8 集成測試【前】
- 9 異步驗證器【前】
- 10 詳解CORS【前】
- 第二節 課程列表
- 第三節 果斷
- 1 初始化【前】
- 2 分頁組件【前】
- 2 分頁組件【前】
- 3 綜合查詢【前】
- 4 綜合查詢【后】
- 4 綜合查詢【后】
- 第節 班級列表
- 第節 教師列表
- 第節 編輯課程
- TODO返回機制【前】
- 4 彈出框組件【前】
- 5 多路由出口【前】
- 第節 刪除課程
- 第七章 權限管理
- 第一節 AOP
- 總結
- 開發規范
- 備用