SpringBoot中數(shù)據(jù)傳輸對象(DTO)的實現(xiàn)
探索在Spring Boot中使用數(shù)據(jù)傳輸對象(DTO)的好處,本文包括了手動創(chuàng)建DTO、使用ModelMapper和Lombok創(chuàng)建DTO的示例。
1. 什么是數(shù)據(jù)傳輸對象(DTO)?
數(shù)據(jù)傳輸對象(DTO)是一種設(shè)計模式,用于封裝和傳輸應(yīng)用程序不同層之間的數(shù)據(jù)。
DTO是輕量級對象,通常只包含必要的字段,不包含任何業(yè)務(wù)邏輯。DTO作用于應(yīng)用程序中不同的業(yè)務(wù)之間的數(shù)據(jù)傳輸,例如在前端和后端之間或在分布式系統(tǒng)中不同的微服務(wù)值之間。
在Spring Boot應(yīng)用程序中,DTO特別有用,因為需要在控制器層、服務(wù)層和持久層之間傳輸數(shù)據(jù)。通過使用DTO就可以將內(nèi)部數(shù)據(jù)模型與外部表示解耦(這點巨好,老鳥都知道),從而更好地控制數(shù)據(jù)傳輸。
2. 在Spring Boot中使用DTO的好處
在Spring Boot應(yīng)用程序中使用DTO有幾個優(yōu)點:
- 數(shù)據(jù)隔離:DTO允許將暴露給外部的數(shù)據(jù)與內(nèi)部的模型隔離。這可以防止暴露敏感和不必要的數(shù)據(jù),并為數(shù)據(jù)交換提供清晰的字段。
- 減少開銷:DTO可以僅包含特定所需的字段,減少網(wǎng)絡(luò)傳輸。這最小化了傳輸大型對象的開銷。
- 版本控制和兼容性:DTO可以使接口支持向后兼容。一個API可以對外提供多個DTO結(jié)構(gòu)。
- 提高安全性:通過對DTO暴露數(shù)據(jù)的控制,就避免了數(shù)據(jù)泄漏以及保護了敏感信息的安全性。
- 增強測試:DTO簡化了單元測試,因為您可以在測試場景中輕松創(chuàng)建和操作它們,而不依賴于復(fù)雜的域?qū)ο蟆?/li>
3. 在Spring Boot中以不同方式使用DTO
3.1. 手動創(chuàng)建DTO
在這種方法中,手動創(chuàng)建對實體結(jié)構(gòu)映射的DTO類。然后編寫代碼在實體對象和DTO之間映射數(shù)據(jù)。
public class UserDTO { private Long id; private String username; private String email; // 構(gòu)造函數(shù)、getter和setter }
創(chuàng)建一個控制器方法演示手動映射:
@RestController @RequestMapping("/api/users") public class UserController { @Autowired private UserService userService; @GetMapping("/{id}") public ResponseEntity getUserById(@PathVariable Long id) { User user = userService.getUserById(id); UserDTO userDTO = new UserDTO(); userDTO.setId(user.getId()); userDTO.setUsername(user.getUsername()); userDTO.setEmail(user.getEmail()); return ResponseEntity.ok(userDTO); } }
輸出:
調(diào)用/api/users/1請求時,將收到UserDTO結(jié)構(gòu)的用戶數(shù)據(jù)。
{ "id": 1, "username": "張三", "email": "zhangsan@qq.com" }
3.2. 使用ModelMapper
ModelMapper用于自動將實體對象映射到DTO,反之亦然。以下是pom.xml依賴:
<dependency> <groupId>org.modelmapper</groupId> <artifactId>modelmapper</artifactId> <version>2.4.3</version> </dependency>
實例化ModelMapper為一個BEAN,這樣就可以在整個程序中使用ModelMapper
以下是創(chuàng)建Spring Boot應(yīng)用程序中ModelMapper Bean的方法:
import org.modelmapper.ModelMapper; import org.springframework.context.annotation.Bean; import org.springframework.context.annotation.Configuration; @Configuration public class MyApplicationConfig { @Bean public ModelMapper modelMapper() { return new ModelMapper(); } }
以下是如何使用ModelMapper將實體對象映射到DTO:
@RestController @RequestMapping("/api/users") public class UserController { @Autowired private UserService userService; @Autowired private ModelMapper modelMapper; // 自動裝配ModelMapper Bean @GetMapping("/{id}") public ResponseEntity getUserById(@PathVariable Long id) { User user = userService.getUserById(id); UserDTO userDTO = modelMapper.map(user, UserDTO.class); // 使用ModelMapper進行映射 return ResponseEntity.ok(userDTO); } }
輸出:
輸出將與手動創(chuàng)建DTO的示例相同。
3.3 使用Lombok
在pom.xml文件中添加Lombok依賴項。
<dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>1.18.22</version> <scope>provided</scope> </dependency>
以下是演示如何使用Lombok為User實體創(chuàng)建DTO。
import lombok.Data; @Data //關(guān)鍵注解 public class UserDTO { private Long id; private String username; private String email; }
以下是如何使用Lombok將實體對象映射到DTO:
@RestController @RequestMapping("/api/users") public class UserController { @Autowired private UserService userService; @GetMapping("/{id}") public ResponseEntity getUserById(@PathVariable Long id) { User user = userService.getUserById(id); UserDTO userDTO = UserDTO.builder() .id(user.getId()) .username(user.getUsername()) .email(user.getEmail()) .build(); return ResponseEntity.ok(userDTO); } }
輸出:
輸出與手動創(chuàng)建DTO的示例相同。
4.在DTO中格式化不同類型的值
在DTO中格式化不同類型的值是確保數(shù)據(jù)在序列化或顯示時以特定格式呈現(xiàn)的常見要求。根據(jù)業(yè)務(wù)要求格式化的值類型,可以使用各種方法,包括注解、自定義方法或外部庫。下面是在DTO中格式化不同類型的值實例:
4.1. 格式化日期和時間
4.1.1. 使用@JsonFormat注解(Jackson)
要在DTO中格式化日期和時間值,可以使用Jackson庫提供的@JsonFormat注解。
import com.fasterxml.jackson.annotation.JsonFormat; public class UserDTO { private Long id; @JsonFormat(pattern = "yyyy-MM-dd HH:mm:ss", timezone = "UTC") private Date registrationDate; // 其他字段、getter和setter }
在此示例中,registrationDateField使用@JsonFormat注解指定所需的日期和時間格式。
4.1.2. 使用SimpleDateFormat(自定義方法)
還可以通過在DTO類中提供自定義getter方法來格式化日期和時間,該方法返回格式化后的日期字符串。
import java.text.SimpleDateFormat; public class UserDTO { private Long id; private Date registrationDate; public String getFormattedRegistrationDate() { SimpleDateFormat sdf = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); return sdf.format(registrationDate); } // 其他字段、getter和setter }
4.2. 格式化數(shù)字
4.2.1. 使用@NumberFormat注解(Spring)
要格式化數(shù)字值,如數(shù)字或貨幣,您可以使用Spring提供的@NumberFormat注解。此注解允許您指定數(shù)字格式模式。
import org.springframework.format.annotation.NumberFormat; public class ProductDTO { private Long id; @NumberFormat(pattern = "#,###.00") private BigDecimal price; // 其他字段、getter和setter }
在此示例中,price字段使用@NumberFormat注解指定數(shù)字格式模式。
4.2.2. 使用DecimalFormat(自定義方法)
還可以通過在DTO類中提供自定義getter方法來格式化數(shù)字,該方法使用DecimalFormat返回格式化后的數(shù)字字符串。
import java.text.DecimalFormat; public class ProductDTO { private Long id; private BigDecimal price; public String getFormattedPrice() { DecimalFormat df = new DecimalFormat("#,###.00"); return df.format(price); } // 其他字段、getter和setter }
4.3. 格式化字符串
4.3.1. 使用自定義方法
對于格式化字符串值,可以在DTO類中創(chuàng)建自定義getter方法以根據(jù)需要操作字符串。例如,您可以優(yōu)化空格、大寫單詞或應(yīng)用任何其他字符串操作邏輯。
public class ArticleDTO { private Long id; private String title; public String getFormattedTitle() { // 自定義格式化邏輯在這里 return title.trim(); // 示例:修剪空格 } // 其他字段、getter和setter }
4.4. 格式化枚舉
4.4.1. 使用自定義方法
在處理DTO中的枚舉時,可以創(chuàng)建自定義getter方法以返回枚舉值的格式化表示形式。例如,您可以將枚舉值轉(zhuǎn)換為大寫或使用不同的表示形式。
public class OrderDTO { private Long id; private OrderStatus status; public String getFormattedStatus() { return status.toString().toUpperCase(); // 示例:轉(zhuǎn)換為大寫 } // 其他字段、getter和setter }
4.5. 格式化布爾值
4.5.1. 使用自定義方法
對于布爾值,可以創(chuàng)建自定義getter方法以返回格式化表示形式,例如“是”或“否”,而不是“true”或“false”。
public class UserDTO { private Long id; private boolean isActive; public String getFormattedIsActive() { return isActive ? "Yes" : "No"; // 示例:轉(zhuǎn)換為"Yes"或"No" } // 其他字段、getter和setter }
通過使用這些方法,可以根據(jù)具體要求格式化DTO中的不同類型的值,確保在序列化或顯示DTO時數(shù)據(jù)以期望的格式呈現(xiàn)。
5.其他注意事項和最佳實踐
5.1. DTO中的驗證
在處理DTO時,考慮數(shù)據(jù)驗證至關(guān)重要。應(yīng)該驗證DTO中的傳入數(shù)據(jù),以確保它滿足所需的約束和業(yè)務(wù)規(guī)則??梢允褂肧pring的驗證注解(如@NotNull、@Size)或自定義驗證注解來驗證DTO字段。
以下是使用Spring的@NotBlank注解進行DTO驗證的示例:
public class UserDTO { @NotNull private Long id; @NotBlank @Size(min = 5, max = 50) private String username; @Email private String email; // 構(gòu)造函數(shù)、getter和setter }
5.2. 用于復(fù)雜嵌套對象的DTO
在實際業(yè)務(wù)中,處理嵌套對象的DTO會很復(fù)雜。需要創(chuàng)建嵌套的DTO來準確映射這些結(jié)構(gòu)。
例如,如果一個User與一組Address對象關(guān)聯(lián),可以創(chuàng)建一個嵌套AddressDTO的UserDTO:
public class UserDTO { private Long id; private String username; private String email; private List<AddressDTO> addresses; // 構(gòu)造函數(shù)、getter和setter }
5.3. DTO版本控制
隨著業(yè)務(wù)的發(fā)展,可能需要對DTO進行更改。為了保持向后兼容性則需要考慮對DTO進行版本控制。例如可以通過向DTO添加版本標識符或在必要時創(chuàng)建新的DTO版本來實現(xiàn)。
5.4. RESTful API中的DTO
DTO在RESTful API中常用于表示客戶端和服務(wù)器之間的數(shù)據(jù)交換。在設(shè)計RESTful端時,應(yīng)仔細選擇或創(chuàng)建DTO,以滿足不同業(yè)務(wù)需求。
6.將DTO與Spring驗證一起使用
Spring提供了一個強大的機制,使用@Valid注解在控制器方法中去驗證DTO。當(dāng)你將DTO參數(shù)與@Valid注解一起使用時,Spring將自動基于DTO類中定義的驗證規(guī)則觸發(fā)驗證。
以下是使用DTO驗證的控制器方法示例:
@RestController @RequestMapping("/api/users") public class UserController { @Autowired private UserService userService; @PostMapping("/create") public ResponseEntity createUser(@Valid @RequestBody UserDTO userDTO) { // 驗證邏輯會自動觸發(fā) // 將UserDTO映射到User實體并保存它 User user = modelMapper.map(userDTO, User.class); User savedUser = userService.saveUser(user); // 返回保存的UserDTO UserDTO savedUserDTO = modelMapper.map(savedUser, UserDTO.class); return ResponseEntity.status(HttpStatus.CREATED).body(savedUserDTO); } }
在此示例中,@Valid注解基于UserDTO類中定義的規(guī)則觸發(fā)驗證。如果驗證失敗,Spring將自動處理驗證錯誤并返回帶有適當(dāng)錯誤消息的響應(yīng)數(shù)據(jù)。
7.微服務(wù)架構(gòu)中的DTO
在微服務(wù)架構(gòu)中,DTO在定義微服務(wù)之間的邊界方面起著關(guān)鍵作用。每個微服務(wù)都可以擁有一套針對其特定需求而量身定制的DTO。這種分離模式確保了微服務(wù)之間的松耦合。
DTO還有助于減少微服務(wù)之間傳輸?shù)臄?shù)據(jù)量,這對于維護基于微服務(wù)的系統(tǒng)的性能和可擴展性至關(guān)重要。
8.結(jié)論
數(shù)據(jù)傳輸對象(DTO)在Spring Boot應(yīng)用程序中不可或缺,它充當(dāng)應(yīng)用程序不同層和外部系統(tǒng)之間的橋梁。通過仔細設(shè)計和使用DTO,可以改善數(shù)據(jù)隔離、減少開銷、增強安全性并簡化測試。無論是選擇手動創(chuàng)建DTO或使用像ModelMapper這樣的庫,還是利用Lombok減少代碼,關(guān)鍵是選擇最適合項目要求和可維護性的方法。將DTO作為架構(gòu)一部分,你將會更好地打造自己健壯高效的Spring Boot應(yīng)用程序。
到此這篇關(guān)于SpringBoot中數(shù)據(jù)傳輸對象(DTO)的實現(xiàn)的文章就介紹到這了,更多相關(guān)SpringBoot 數(shù)據(jù)傳輸對象內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java 中 getClass() 方法的使用與原理深入分析(對象類型信息)
在 Java 編程中,getClass() 是一個非常重要的方法,它用于獲取對象的運行時類信息,無論是調(diào)試代碼、反射操作,還是類型檢查,getClass() 都扮演著關(guān)鍵角色,本文將深入探討 getClass() 的使用方法、底層原理以及實際應(yīng)用場景,感興趣的朋友一起看看吧2024-12-12Spring MVC學(xué)習(xí)教程之視圖深入解析
這篇文章主要給大家介紹了關(guān)于Spring MVC學(xué)習(xí)教程之視圖解析的相關(guān)資料,文中通過示例代碼介紹的非常詳細,對大家學(xué)習(xí)或使用spring mvc具有一定的參考學(xué)習(xí)價值,需要的朋友們下面來一起看看吧2018-11-11詳解tryAcquire()、addWaiter()、acquireQueued()
這篇文章主要tryAcquire()、addWaiter()、acquireQueued()的用法,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03idea中文件被Mark as Plain Text后恢復(fù)方式
在IntelliJ IDEA中,如果錯誤地將文件標記為純文本(Mark as Plain Text),可以通過在項目目錄中右鍵點擊文件并選擇“Mark as”來恢復(fù)原文件類型2024-11-11