Java中的 VO,BO,DO 對象命名問題小結(jié)
最近,有小伙伴反饋:很困惑代碼里用什么對象來傳輸數(shù)據(jù),是共用一個(gè)對象還是每層各用一個(gè)對象,對于數(shù)據(jù)對象又該如何命名,種種困惑導(dǎo)致代碼結(jié)構(gòu)很混亂。針對該問題,今天就一起來聊聊這個(gè)看似簡單,其實(shí)很多人都在誤用的代碼分層以及對象命名問題。
說起代碼分層,那就不得不提起 MVC模式,相信很多程序員的編程啟蒙階段都有它的陪伴。
1. 什么是MVC?
MVC是后端業(yè)務(wù)開發(fā)最常見,使用頻率最高的一種編程模式,因此,本文將結(jié)合 Java語言對 MVC做簡要的介紹:
M:Mode,數(shù)據(jù)層,負(fù)責(zé)和數(shù)據(jù)庫交互;
V:View,視圖層,負(fù)責(zé)顯示數(shù)據(jù)給用戶,并向用戶呈現(xiàn)信息;
C:Controller,邏輯層,負(fù)責(zé)處理用戶的輸入和相應(yīng)操作;
早期,在 Java語言中,JSP(Java Server Page) 是 View視圖的最好體現(xiàn),有過 JSP經(jīng)歷的開發(fā)人員一定深深體會到了它的辛酸,隨著互聯(lián)網(wǎng)的快速發(fā)展,這種前端頁面在后端代碼中去編寫顯然滿足不了市場的需求,隨之而來的是前后端分離,基于前后端分離的場景,MVC模式與 Java常見的代碼分層的對應(yīng)關(guān)系可以如下圖所示:
上圖 Controller-Service-Repository 三層架構(gòu)模型是前后端分離場景下 MVC最經(jīng)典體現(xiàn),目前,絕大多數(shù)的公司都是直接或者間接使用這個(gè)模型。盡管不同公司,代碼架構(gòu)略有差異,但萬變不離窮,掌握了MVC三層架構(gòu)就能輕松的去使用其他模型。
2. 公共對象貫穿MVC
講完了 MVC模型,接下來的問題是:Controller-Service-Repository 各層之間通過什么對象傳輸數(shù)據(jù)呢?
你是否寫過這樣的代碼?一個(gè)公共的對象,貫穿整個(gè)MVC,如下圖, User對象從Controller層接收用戶參數(shù),一直到Repository 層,最終存儲到 DB中。
上圖是很多中初級程序員最容易出現(xiàn)的代碼,一個(gè)類對象將參數(shù)從最頂層傳到最底層,然后又將最底層的 DB數(shù)據(jù)輸出到最頂層,簡單粗暴,這樣會產(chǎn)生什么問題呢?
- 安全性:用戶的入?yún)⒖梢灾眰鞯紻B,存在SQL注入到風(fēng)險(xiǎn),DB出來的數(shù)據(jù)直接到達(dá)View層,這樣某些敏感數(shù)據(jù)可能會泄漏,比如注冊功能,可能把密碼暴露。
- 使用困惱:對于接收入?yún)⒌膶ο螅话闶切枰裁磾?shù)據(jù),定義什么參數(shù),假如把一個(gè)大而全的對象直接暴露給用戶,這樣對接的用戶對于入?yún)髦稻蜁芾Щ蟆?/li>
- 耦合性:如果各層都依賴于相同的數(shù)據(jù)對象,更改一個(gè)層次的數(shù)據(jù)結(jié)構(gòu)可能會影響其他層次,系統(tǒng)耦合性太強(qiáng)。
這里只列舉了 3個(gè)比較常見的問題,那么有什么好的方式可以解決這個(gè)問題呢?
最常用的方法是:使用VO,BO,DO對各層進(jìn)行數(shù)據(jù)傳輸。
3. VO,BO,DO
VO:View Object,視圖對象,用于 Controller層的數(shù)據(jù)對象;
BO:Business Object,業(yè)務(wù)對象,用于 Service層的數(shù)據(jù)對象;
DO:Domain Object,數(shù)據(jù)對象,用于 Repository層的數(shù)據(jù)對象,也可以叫做 Entity;
VO,BO,DO本質(zhì)上是對數(shù)據(jù)類對象起一個(gè)規(guī)范的名字,比如:UserVO,UserBO,UserDO,它就和 xxxController,xxxService,xxxRepository 一樣,見名知意,讓人看見名字就知道它的職責(zé)。
為了更好的說明 VO,BO,DO,我們以一個(gè)生活中的實(shí)例“領(lǐng)導(dǎo)配秘書“來進(jìn)行講解。如下圖:
當(dāng)各層共用一個(gè)數(shù)據(jù)對象時(shí),類比 董事長,CEO,總經(jīng)理共用一個(gè)秘書,我們一起來分析上面提到的 3個(gè)常見問題:
- 安全性:共用秘書,董事長的秘密會不會暴露給CEO和總經(jīng)理呢?安全嗎?
- 耦合性:一個(gè)秘書要干 3個(gè)人的事情,每個(gè)領(lǐng)導(dǎo)的事情不一樣,怎么解耦?
- 使用困惑:當(dāng)員工要反映問題時(shí),秘書該反映到哪一層領(lǐng)導(dǎo)呢?
因此,聞道有先后,術(shù)業(yè)有專攻,專業(yè)的事就得干專業(yè)的人來干。董事長,CEO,總經(jīng)理共用一個(gè)秘書顯然不合理,必須配備自己的專職秘書;同理,Controller層,Service層,Repository層共用一個(gè)數(shù)據(jù)類對象也是不合理的,必須配備特定的類對象,對比圖如下:
通過上圖中每層領(lǐng)導(dǎo)配備專職秘書的實(shí)例,是不是能很好的理解 VO,BO,DO的作用:各層對象只負(fù)責(zé)自己本層的數(shù)據(jù),劃清邊界,清晰職責(zé)。
因此,MVC,代碼分層以及各層數(shù)據(jù)對象的對應(yīng)關(guān)系可以描述成下圖:
需要注意:因?yàn)?VO,BO,DO 需要定義屬性來傳遞數(shù)據(jù),因此難免會帶來一個(gè)問題:三者出現(xiàn)重復(fù)的代碼。這個(gè)問題合理嗎?需要如何處理呢?
4. 消除重復(fù)代碼
首先我們來分析三者出現(xiàn)重復(fù)的代碼是否合理?
比如,用戶注冊功能,我們會在 UserVO,UserBO,UserDO中都定義相同的字段 username, password,如下圖:
盡管三個(gè)類中都包含了相同的字段和getter和setter方法,但是因?yàn)樗麄兊恼Z義不一樣,職責(zé)也不一樣,所以這種重復(fù)是合理的。
有沒有好的辦法來消除這種重復(fù)呢?
有,將相同的屬性抽到公共的類中,通過繼承獲得,如下圖:
上圖,我們抽取了一個(gè)父類 BaseUser來存放共用的字段,這樣就可以達(dá)到代碼復(fù)用,然后在每個(gè)子類對象中去定義特定的屬性。既然類對象之間需要傳輸數(shù)據(jù),那么該如何相互轉(zhuǎn)換呢?
5. 對象相互轉(zhuǎn)換
在實(shí)際開發(fā)中,我們一般會定義 Converter 類來負(fù)責(zé)對象之間的拷貝,如下圖:
方法一:手動set
該方式缺點(diǎn):需要手動編寫 set方法,代碼量比較多;優(yōu)點(diǎn):一個(gè)類對象更換了字段名,賦值會報(bào)錯(cuò),能及時(shí)發(fā)現(xiàn)。
方法二:使用Apache的BeanUtils等工具進(jìn)行拷貝
該方式優(yōu)點(diǎn):使用三方類包裝的方式,簡單易用;缺點(diǎn):像BeanUtils工具,只會拷貝相同的屬性,當(dāng)一個(gè)類對象換了字段名,不會報(bào)錯(cuò),不能及時(shí)發(fā)現(xiàn)。
兩種方式各有優(yōu)缺點(diǎn),實(shí)際開發(fā)中都會用到,具體選擇哪一種需要技術(shù)團(tuán)隊(duì)內(nèi)部討論決定。
6. 總結(jié)
- 本文講解了 MVC模式以及Java與之對應(yīng)的代碼分層;
- 本文講解了在代碼各層共用一個(gè)類對象傳輸數(shù)據(jù)的優(yōu)缺點(diǎn),并且結(jié)合現(xiàn)實(shí)生活中領(lǐng)導(dǎo)配秘書來類比講解;
- 本文講解VO,BO,DO 的作用以及如何使用,分析了如何消除三者之間重復(fù)的代碼,同樣結(jié)合現(xiàn)實(shí)生活中領(lǐng)導(dǎo)配秘書來類比講解;
- MVC 是代碼分層的精髓,盡管很多公司不一定嚴(yán)格遵守MVC,但是萬變不離其宗,掌握了精髓,才能輕松玩轉(zhuǎn)其他的分層風(fēng)格;
到此這篇關(guān)于Java中的 VO,BO,DO 對象命名問題的文章就介紹到這了,更多相關(guān)Java VO 對象命名內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
利用5分鐘快速搭建一個(gè)springboot項(xiàng)目的全過程
Spring Boot的監(jiān)控能夠使開發(fā)者更好地掌控應(yīng)用程序的運(yùn)行狀態(tài),下面這篇文章主要給大家介紹了關(guān)于如何利用5分鐘快速搭建一個(gè)springboot項(xiàng)目的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-05-05Java中的throws關(guān)鍵字處理異常的最佳實(shí)踐記錄
在Java編程中,異常處理是保證程序健壯性和穩(wěn)定性的重要手段,除了使用try-catch塊捕獲異常外,Java還提供了throws關(guān)鍵字,允許我們將異常拋給調(diào)用者處理,本文介紹Java中的throws關(guān)鍵字處理異常的最佳實(shí)踐記錄,感興趣的朋友一起看看吧2025-01-01Spring Security 構(gòu)建rest服務(wù)實(shí)現(xiàn)rememberme 記住我功能
這篇文章主要介紹了Spring Security 構(gòu)建rest服務(wù)實(shí)現(xiàn)rememberme 記住我功能,需要的朋友可以參考下2018-03-03Mybatis?selectKey 如何返回新增用戶的id值
這篇文章主要介紹了Mybatis?selectKey 如何返回新增用戶的id值,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-01-01Java使用POI導(dǎo)出Excel(一):單sheet
這篇文章介紹了Java使用POI導(dǎo)出Excel的方法,文中通過示例代碼介紹的非常詳細(xì)。對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2022-10-10Security6.4.2?自定義異常中統(tǒng)一響應(yīng)遇到的問題
本文主要介紹了Security6.4.2?自定義異常中統(tǒng)一響應(yīng)遇到的問題,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2025-03-03