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

如果跳過 DTO 會發生什麼

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

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]刪除
最新教學 更多>
  • 棄用 `ereg_replace` 後,如何將多個空格替換為單一空格?
    棄用 `ereg_replace` 後,如何將多個空格替換為單一空格?
    用單一空格取代多個空格:棄用ereg_replace雖然使用ereg_replace 以單一空格取代多個空格看起來很簡單,但它已被棄用。因此,您在嘗試使用它時可能會遇到錯誤。本文提出了一種替代解決方案。 遷移到 preg_replace()要取代 ereg_replace,請切換到 preg_rep...
    程式設計 發佈於2024-11-06
  • 如何開始自由工作?
    如何開始自由工作?
    您是否正在努力透過 Upwork 賺錢?別擔心!我曾經去過那裡,我來這裡是為了分享我將這些掙扎轉化為成功的旅程。憑藉奉獻精神和正確的方法,您可以開始在 Upwork 上賺錢,並打開通往一系列機會的大門。讓我們深入探討最大化您的收入潛力的關鍵步驟。 掌握技巧遊戲 在 Upwork 上...
    程式設計 發佈於2024-11-06
  • 如何將 8 個字元作為打包單精度浮點數載入到 __m256 變數中?
    如何將 8 個字元作為打包單精度浮點數載入到 __m256 變數中?
    將8 個字元從記憶體載入到__m256 變數中作為打包單精度浮點數為了優化高斯模糊演算法,您尋求用__m256 內在變數取代浮點緩衝區的使用。本問題旨在確定此任務的最佳指令。 AVX2 架構說明:利用 PMOVZX 將字元零擴展為 32 位元256b 暫存器中的整數。 使用 VCVTDQ2PS 就地...
    程式設計 發佈於2024-11-06
  • 如何在Python中尋找字串中子字串的第n次出現?
    如何在Python中尋找字串中子字串的第n次出現?
    查找字串中某個子字串第n 次出現識別某個子字串第n 次出現對應的索引為各種程式場景中經常出現的任務。在Python中,沒有專門為此目的而設計的內建函數。但是,可以採用多種方法來實現此結果。 一種簡單的方法是使用循環來迭代字串併計算子字串的出現次數。起始索引被初始化為第一次出現的結果,循環繼續直到找到...
    程式設計 發佈於2024-11-06
  • 程式語言解釋
    程式語言解釋
    ? 注意:缩略图是使用生成的 ComfyUI 帮助下的 Flux Schnell 模型; 本文是在 NI - 自然智能 的帮助下撰写的 ? 没有时间阅读? - 观看我们创建的视频: 集会 ? 最初所有程序都是用二进制编写的 - 所谓的汇编 编程语言。你从字面上指示CPU在内做什么 ...
    程式設計 發佈於2024-11-06
  • Go Context — TODO() 與 Background() 不再令人困惑!
    Go Context — TODO() 與 Background() 不再令人困惑!
    在 Go 中,上下文包有助於管理請求範圍的值、取消訊號和截止日期。 啟動上下文的兩種常見方法是 context.TODO() 和 context.Background()。 儘管它們的行為相似,但它們的目的不同。 上下文.背景() 當您不需要任何特殊處理(例如取消或截止日期...
    程式設計 發佈於2024-11-06
  • 如何偵測 CMake 中的 C++11 編譯器支援?
    如何偵測 CMake 中的 C++11 編譯器支援?
    CMake 中C 11 編譯器支援的檢測概述在本指南中,我們探討自動偵測編譯器是否支援C 11 的方法CMake,提供對最新和以前CMake 版本的全面分析。 CMake 3.1.0 及更高版本CMake 3.1.0 版本引進了一個強大的功能:偵測編譯器。這是透過cmake_minimum_requ...
    程式設計 發佈於2024-11-06
  • 基於屬性的測試:深入探討現代測試方法
    基於屬性的測試:深入探討現代測試方法
    基于属性的测试是一种强大的测试方法,它侧重于软件的属性或特征,而不是特定的输入输出情况。与手动定义特定测试用例的传统测试不同,基于属性的测试会自动生成各种输入来验证某些属性是否始终成立。这种方法可以更广泛地探索潜在场景,使其成为发现隐藏错误并确保稳健的软件行为的有效方法。 测试方法的演变 从传统的...
    程式設計 發佈於2024-11-06
  • 為開發人員和安全團隊提供主動的 AppSec 持續漏洞管理
    為開發人員和安全團隊提供主動的 AppSec 持續漏洞管理
    现代软件开发环境中哪些日益增长的网络安全风险让 CISO 忙碌? 开发人员和安全团队面临着越来越多的威胁,从复杂的开源和供应商控制的供应链攻击到 AI 生成的代码引入的漏洞,例如提示注入和 GitHub Copilot 的代码安全性差。现代应用程序通常严重依赖开源组件(例如在 npm、PyPI 或 ...
    程式設計 發佈於2024-11-06
  • 如何使用 React 對 MeteorJS 中的 Bootstrap Spacing 實用程式類別進行故障排除?
    如何使用 React 對 MeteorJS 中的 Bootstrap Spacing 實用程式類別進行故障排除?
    在 Bootstrap 中使用間距實用程式類別在 Bootstrap 中,間距實用程式類別可讓您輕鬆控制元素周圍的間距。但是,如果您在使用它們時遇到問題,這裡有一個指南可以幫助您解決。 更新的間距語法(Bootstrap 4 和 5)Bootstrap 4引入了間距實用程式類別的簡化語法:邊距:m{...
    程式設計 發佈於2024-11-06
  • 如何在Python中設定子程序的工作目錄?
    如何在Python中設定子程序的工作目錄?
    如何在Python中設定子程序的工作目錄在Python中,subprocess.Popen()函數允許您在Py thon中執行指令子程序。一個常見的要求是指定子程序的工作目錄。 問題:如何使用 subprocess.Popen() 設定子程序的工作目錄? 答案:要指定工作目錄,請使用 subproc...
    程式設計 發佈於2024-11-06
  • Pandas 什麼時候創建視圖而不是副本?
    Pandas 什麼時候創建視圖而不是副本?
    Pandas 視圖與副本生成規則Pandas 在決定 DataFrame 上的切片操作是否產生視圖或結果時採用特定規則複製。透過了解這些規則,您可以優化資料操作並避免意外行為。 從始終產生副本的操作開始:所有操作,除了那些專門設計用於修改的操作就地 DataFrame,創建副本。 只有某些操作支援 ...
    程式設計 發佈於2024-11-06
  • 使用代理伺服器解鎖地理限制網站
    使用代理伺服器解鎖地理限制網站
    利用代理服务器绕过区域封锁是一种常用且有效的方法。代理服务器作为中介,可以隐藏用户的真实IP地址,使用户的请求看起来像是来自代理服务器的地理位置,从而绕过区域封锁。 使用代理服务器绕过区域封锁的关键步骤:‌‌ 选择合适的代理服务器‌:根据目标区域的网络环境和遮挡情况,选择覆盖该区域的...
    程式設計 發佈於2024-11-06
  • 如何為三角形中的線性漸變鋸齒線建立平滑邊緣?
    如何為三角形中的線性漸變鋸齒線建立平滑邊緣?
    為線性漸變鋸齒線創建平滑邊緣為了設計具有由兩個三角形形成的尖底的響應式圖像,開發人員在三角形線上遇到了意外的鋸齒狀邊緣。為了解決這個問題,我們探索了產生更平滑漸變過渡的策略。 雖然硬停止線性漸變影像中的顏色通常會導致鋸齒狀邊緣,但調整停止點和起始點可以緩解此問題。不要突然從一種顏色變為另一種顏色,而...
    程式設計 發佈於2024-11-06
  • Java 中「static」的魔力:一為所有,一切為一!
    Java 中「static」的魔力:一為所有,一切為一!
    老实说,当我们第一次遇到 static 关键字时,我们都会想:“这是什么魔法?” ?但别担心,我会用一种简单、深入、甚至有点有趣的方式来分解它! 想象一下你正在参加一个聚会?你和你所有的朋友都戴着帽子。但每个人都必须分享一顶帽子。这基本上就是 Java 中 static 关键字的作用!您不必为每个朋...
    程式設計 發佈於2024-11-06

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

Copyright© 2022 湘ICP备2022001581号-3