」工欲善其事,必先利其器。「—孔子《論語.錄靈公》
首頁 > 程式設計 > 如果跳過 DTO 會發生什麼

如果跳過 DTO 會發生什麼

發佈於2024-08-02
瀏覽:312

What can happen if you skip the DTOs

像SpringBoot這樣的框架可以為你做很多事情,這真是太好了。

您只需要一個 JPA 實體類別加上一個簡單的儲存庫接口,SpringData 將為您提供典型 CRUD 資料庫操作所需的一切。

您編寫了一個簡單的 REST 控制器類,並且運行了一個 REST API,對吧?

嘿,但你忘了寫 DTO !但是,當您的應用程式無需它也可以運行時,為什麼您實際上需要它?

當然有一些一般原因:

  • 分層結構(例如六角形架構或連接埠和適配器):為了可維護性,最好將外部通訊程式碼與核心(業務邏輯)解耦
  • 安全與效能:如果您按原樣在API 中公開資料庫結構,您很快就會達到公開超出需求的程度;可能被惡意行為者濫用或浪費資源(CPU、記憶體和網路頻寬)
  • DTO 與 JPA 實體不同,可以是不可變的(您可以使用 Java 記錄),這有利於資料驅動(函數式)程式設計風格、良好的單元測試、更安全的並發性等。

但是其他奇怪的事情也可能發生。我將根據我的經驗向您展示一個奇怪的例子。

此 GitHub 儲存庫包含一個無需 DTO 即可運行的簡單應用程式。有一個User實體,每個User可以有多個Transaction。我們甚至在儲存庫和 RestController 之間有一個 Service bean,用於捕獲可能的資料庫存取異常。

由於我們想要製作一個可用於生產的應用程序,因此我們不希望 Hibernate 產生 DDL。相反,我們有一個 schema.sql 來建立表(稍後我們可能會切換到 Flyway 或 Liquibase)。對於我們的簡單範例,我們還有一個 data.sql,以便我們的表不為空。

當我們執行應用程式並呼叫 http://localhost:8080/users 處的 API 端點時,我們會得到包含使用者及其交易的預期 JSON。

現在我們專注於Transaction類別中的兩行程式碼,標記為//!!

@JsonIgnore //!!

第一個味道是,在 Transaction 類別中,我們必須將 @JsonIgnore 註解加入 User 參考中。如果沒有該註釋,JSON 序列化會因無限遞歸而崩潰。

現在讓我們想像一下,有人在向 Transaction 實體新增另一個欄位(描述)時犯了一個錯誤,但忘記調整 SQL 語句(或在尚未應用架構變更的環境中執行應用程式)。

私有字串描述;//!!

當然,現在API呼叫失敗了。但看看錯誤處理! UserService 內的 catch 子句未如預期運作。相反,我們可以在日誌中看到奇怪的堆疊追蹤:
GlobalExceptionHandler:意外錯誤org.springframework.http.converter.HttpMessageNotWritableException:無法寫入JSON:

我曾經見過這種情況(顯然,應用程式比這個範例大得多),我花了很長時間才理解為什麼 SQL 異常逃逸了服務以及為什麼我收到 HttpMessageNotWritableException。你能看見它嗎?

發生的情況是,UserService 類別(透過 UserRepository)僅查詢 USERS 資料庫表。由於預設的 Hibernate 延遲加載,事務實體不是結果的一部分。只有當 Jackson 反序列化器嘗試從 User 實例建立 JSON 時,它才會呼叫其 getTransactions 方法,使 Hibernate 取得 Transaction 實體。

這就是為什麼我們得到一個奇怪的堆疊跟踪,結合了 JSON 和 SQL 的內容。這個例外被 GlobalExceptionHandler 捕獲,但不知道如何處理它,這就是為什麼日誌訊息為「意外錯誤」的原因。

我希望這個小練習能讓您更深入地了解允許應用程式的不同層混合是多麼危險。在應用程式還很小的時​​候只看到應用程式的「晴天」場景可能會導致一些開發人員繼續做錯誤的事情,直到為時已晚。

您不必編寫在 DTO 和應用程式的其他層之間映射欄位的樣板程式碼。 MapStruct 可以為您做到。

版本聲明 本文轉載於:https://dev.to/marianvarga/what-c​​an-happen-if-you-skip-the-dtos-aaj?1如有侵犯,請聯絡[email protected]刪除
最新教學 更多>
  • 在 Go 中使用 WebSocket 進行即時通信
    在 Go 中使用 WebSocket 進行即時通信
    构建需要实时更新的应用程序(例如聊天应用程序、实时通知或协作工具)需要一种比传统 HTTP 更快、更具交互性的通信方法。这就是 WebSockets 发挥作用的地方!今天,我们将探讨如何在 Go 中使用 WebSocket,以便您可以向应用程序添加实时功能。 在这篇文章中,我们将介绍: WebSoc...
    程式設計 發佈於2024-12-27
  • HTML 格式標籤
    HTML 格式標籤
    HTML 格式化元素 **HTML Formatting is a process of formatting text for better look and feel. HTML provides us ability to format text without us...
    程式設計 發佈於2024-12-27
  • 插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入資料時如何修復「常規錯誤:2006 MySQL 伺服器已消失」?
    插入記錄時如何解決「一般錯誤:2006 MySQL 伺服器已消失」介紹:將資料插入MySQL 資料庫有時會導致錯誤「一般錯誤:2006 MySQL 伺服器已消失」。當與伺服器的連線遺失時會出現此錯誤,通常是由於 MySQL 配置中的兩個變數之一所致。 解決方案:解決此錯誤的關鍵是調整wait_tim...
    程式設計 發佈於2024-12-27
  • Bootstrap 4 Beta 中的列偏移發生了什麼事?
    Bootstrap 4 Beta 中的列偏移發生了什麼事?
    Bootstrap 4 Beta:列偏移的刪除和恢復Bootstrap 4 在其Beta 1 版本中引入了重大更改柱子偏移了。然而,隨著 Beta 2 的後續發布,這些變化已經逆轉。 從 offset-md-* 到 ml-auto在 Bootstrap 4 Beta 1 中, offset-md-*...
    程式設計 發佈於2024-12-27
  • 儘管程式碼有效,為什麼 POST 請求無法擷取 PHP 中的輸入?
    儘管程式碼有效,為什麼 POST 請求無法擷取 PHP 中的輸入?
    解決PHP 中的POST 請求故障在提供的程式碼片段中:action=''而非:action="<?php echo $_SERVER['PHP_SELF'];?>";?>"檢查$_POST陣列:表單提交後使用 var_dump 檢查 $_POST 陣列的內...
    程式設計 發佈於2024-12-27
  • 大批
    大批
    方法是可以在物件上呼叫的 fns 數組是對象,因此它們在 JS 中也有方法。 slice(begin):將陣列的一部分提取到新數組中,而不改變原始數組。 let arr = ['a','b','c','d','e']; // Usecase: Extract till index ...
    程式設計 發佈於2024-12-27
  • 如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    如何在 PHP 中組合兩個關聯數組,同時保留唯一 ID 並處理重複名稱?
    在 PHP 中組合關聯數組在 PHP 中,將兩個關聯數組組合成一個數組是常見任務。考慮以下請求:問題描述:提供的代碼定義了兩個關聯數組,$array1和$array2。目標是建立一個新陣列 $array3,它合併兩個陣列中的所有鍵值對。 此外,提供的陣列具有唯一的 ID,而名稱可能重疊。要求是建構一...
    程式設計 發佈於2024-12-27
  • 為什麼 C 和 C++ 忽略函式簽章中的陣列長度?
    為什麼 C 和 C++ 忽略函式簽章中的陣列長度?
    將陣列傳遞給C 和C 中的函數問題:為什麼C和C 編譯器允許在函數簽章中宣告數組長度,例如int dis(char a[1])(當它們不允許時)強制執行? 答案:C 和C 中用於將數組傳遞給函數的語法是歷史上的奇怪現象,它允許將指針傳遞給第一個元素詳細說明:在C 和C 中,數組不是透過函數的引用傳遞...
    程式設計 發佈於2024-12-26
  • 如何刪除 MySQL 中的重音符號以改進自動完成搜尋?
    如何刪除 MySQL 中的重音符號以改進自動完成搜尋?
    在MySQL 中刪除重音符號以實現高效的自動完成搜尋管理大型地名資料庫時,確保準確和高效至關重要資料檢索。使用自動完成功能時,地名中的重音可能會帶來挑戰。為了解決這個問題,一個自然的問題出現了:如何在 MySQL 中刪除重音符號以改善自動完成功能? 解決方案在於為資料庫列使用適當的排序規則設定。透過...
    程式設計 發佈於2024-12-26
  • 如何在MySQL中實作複合外鍵?
    如何在MySQL中實作複合外鍵?
    在 SQL 中實作複合外鍵一個常見的資料庫設計涉及使用複合鍵在表之間建立關係。複合鍵是多個列的組合,唯一標識表中的記錄。在這個場景中,你有兩個表,tutorial和group,你需要將tutorial中的複合唯一鍵連結到group中的欄位。 根據MySQL文檔,MySQL支援外鍵對應到複合鍵。但是,...
    程式設計 發佈於2024-12-26
  • 為什麼我的 JComponent 隱藏在 Java 的背景圖片後面?
    為什麼我的 JComponent 隱藏在 Java 的背景圖片後面?
    調試背景圖像隱藏的JComponent在Java 應用程式中使用JComponent(例如JLabels)時,必須確保正確的行為和可見度。如果遇到組件隱藏在背景圖像後面的問題,請考慮以下方法:1。正確設定組件透明度:確保背景面板是透明的,以允許底層組件透過。使用setOpaque(false)方法來...
    程式設計 發佈於2024-12-26
  • 如何在 PHP 中轉換所有類型的智慧引號?
    如何在 PHP 中轉換所有類型的智慧引號?
    在 PHP 中轉換所有類型的智慧引號智慧引號是用來取代常規直引號(' 和")的印刷標記。它們提供了更精緻和然而,軟體應用程式通常會在不同類型的智能引號之間進行轉換,從而導致不一致。智能引號中的挑戰轉換轉換智慧引號的困難在於用於表示它們的各種編碼和字符,不同的作業系統和軟體程式採用自...
    程式設計 發佈於2024-12-26
  • 循環 JavaScript 陣列有哪些不同的方法?
    循環 JavaScript 陣列有哪些不同的方法?
    使用 JavaScript 迴圈遍歷陣列遍歷陣列的元素是 JavaScript 中常見的任務。有多種方法可供選擇,每種方法都有自己的優點和限制。讓我們探討一下這些選項:陣列1。 for-of 遵循(ES2015 )此循環使用迭代器迭代數組的值:const arr = ["a", ...
    程式設計 發佈於2024-12-26
  • 如何在 Python 中有效地暫停 Selenium WebDriver 執行?
    如何在 Python 中有效地暫停 Selenium WebDriver 執行?
    Selenium WebDriver 中的等待與條件語句問題: 如何在 Python 中暫停 Selenium WebDriver 執行幾毫秒? 答案:雖然time.sleep() 函數可用於暫停執行指定的秒數,在 Selenium WebDriver 自動化中一般不建議使用。 使用 Seleniu...
    程式設計 發佈於2024-12-26
  • C++ 賦值運算子應該是虛擬的嗎?
    C++ 賦值運算子應該是虛擬的嗎?
    C 中的虛擬賦值運算子及其必要性雖然賦值運算子可以在C 中定義為虛擬,但這不是強制要求。然而,這種虛擬聲明引發了關於虛擬性的必要性以及其他運算子是否也可以虛擬的問題。 虛擬賦值運算子的案例賦值運算子本質上並非虛擬。然而,當將繼承類別的物件分配給基類變數時,它就變得必要了。這種動態綁定保證了呼叫基於物...
    程式設計 發佈於2024-12-26

免責聲明: 提供的所有資源部分來自互聯網,如果有侵犯您的版權或其他權益,請說明詳細緣由並提供版權或權益證明然後發到郵箱:[email protected] 我們會在第一時間內為您處理。

Copyright© 2022 湘ICP备2022001581号-3