亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb

首頁 > 開發 > Java > 正文

使用Spring Data JDBC實現DDD聚合的示例代碼

2024-07-14 08:42:27
字體:
來源:轉載
供稿:網友

本文討論了Spring Data JDBC如何實現DDD中聚合根存儲的設計思路,其中主要討論了是不是每個實體都需要一個對應數據表,這種問題需要根據具體情況而定。

Spring Data JDBC比JPA更容易理解,比如對象引用特性會很有趣。作為第一個示例,請考慮以下領域模型:

class PurchaseOrder { private @Id Long id; private String shippingAddress; private Set<OrderItem> items = new HashSet<>(); void addItem(int quantity, String product) {  items.add(createOrderItem(quantity, product)); } private OrderItem createOrderItem(int quantity, String product) {  OrderItem item = new OrderItem();  item.product = product;  item.quantity = quantity;  return item; }}class OrderItem { int quantity; String product;}

另外,考慮如下定義的存儲庫:

interface OrderRepository extends CrudRepository<PurchaseOrder, Long> { @Query("select count(*) from order_item") int countItems();}

如果使用商品創建訂單,希望所有商品都能保存:

@Autowired OrderRepository repository;@Testpublic void createUpdateDeleteOrder() { PurchaseOrder order = new PurchaseOrder(); order.addItem(4, "Captain Future Comet Lego set"); order.addItem(2, "Cute blue angler fish plush toy"); PurchaseOrder saved = repository.save(order); assertThat(repository.count()).isEqualTo(1); assertThat(repository.countItems()).isEqualTo(2); …

此外,如果刪除PurchaseOrder,它的所有項目也應該被刪除。

 … repository.delete(saved); assertThat(repository.count()).isEqualTo(0); assertThat(repository.countItems()).isEqualTo(0);}

如果我們需要一個語法上相同但語義上不同的關系呢?上述訂單中包含訂單條目OrderItem , 當訂單刪除時,包含的OrderItem 都刪除了,但是看看看看下面案例,也是使用包含一個集合:

class Book { // … Set<Author> authors = new HashSet<>();}

當書籍絕版時,將Book刪除。所有的作者Author也都丟失了。這當然不是你想要的,因為一些作者可能也寫過其他書。

怎么辦?

讓我們看一看存儲庫實際存在的內容。這與一遍又一遍的問題密切相關:是否應該在JPA中為每個表創建一個存儲庫?

而正確和權威的答案是“不”。存儲庫持久聚合并加載聚合。聚合是一個包含各種對象的群,它應始終保持一致。此外,它應始終保持(和加載)在一起。它有一個對象,稱為聚合根,它是唯一允許外部訪問或引用聚合內部的代理或管理者。聚合根是傳遞給存儲庫的,以便持久化聚合里面的對象群。

這提出了一個問題:Spring Data JDBC如何確定什么是聚合的一部分,哪些不是?答案非常簡單:非瞬態non-transient 引用都是聚合的一部分,這樣就可從聚合根到達聚合內部所有內容。

OrderItem實例是聚合的一部分,因此被刪除; Author正好相反,實例不是Book聚合的一部分,因此不應刪除。所以不應該從Book內部去引用那些作者Author對象。

問題解決了。好吧,......不是真的。我們仍然需要存儲和訪問有關Book和Author之間的關系信息。答案可以在領域驅動設計(DDD)中找到,它建議使用ID而不是直接引用。這適用于各種多對X關系。

如果多個聚合引用同一個實體,則該實體不能成為引用它的多個聚合的一部分,因為它只能是其中一個聚合的一部分。因此,任何“多對一”和“多對多”關系都只能通過引用id來建模實現了。

這樣可以實現多種目的:

1. 清楚地表示了聚合的邊界。

2. 還完全解耦(至少在應用程序的領域模型中)所涉及的兩個聚合。

3. 這種分離可以用不同的方式在數據庫中表示:

a. 以通常的方式保留數據庫,包括所有外鍵。這意味著必須確保以正確的順序創建和保留聚合。

b. 使用延遲約束,僅在事務的提交階段進行檢查。這可能會提高吞吐量。它還編纂了最終一致性的版本,其中“最終”與交易結束相關聯。這也允許引用從未存在的聚合,只要它僅在事務期間發生。這對于避免大量基礎結構代碼只是為了滿足外鍵和非空約束可能是有用的。

c. 完全刪除外鍵,實現真正的最終一致性。

d. 將引用的聚合保留在不同的數據庫中,甚至可能是No SQL存儲。

無論如何,即使Spring Data JDBC也鼓勵應用模塊化。此外,如果嘗試遷移一個具有10年歷史的單體,你就會明白它的價值。

使用Spring Data JDBC,您可以建模多對多關系,如下所示:

class Book { private @Id Long id; private String title; private Set<AuthorRef> authors = new HashSet<>(); public void addAuthor(Author author) {  authors.add(createAuthorRef(author)); } private AuthorRef createAuthorRef(Author author) {  Assert.notNull(author, "Author must not be null");  Assert.notNull(author.id, "Author id, must not be null");  AuthorRef authorRef = new AuthorRef();  authorRef.authorId = author.id;  return authorRef; }}@Table("Book_Author")class AuthorRef { Long authorId ;}class Author { @Id Long id; String name;}

請注意額外的類:AuthorRef,它表示有關某個作者的Book聚合的知識。它可能包含有關作者的其他聚合信息,然后實際上會在數據庫中重復。考慮到Author數據庫可能與Book數據庫完全不同,這會產生很多問題。

另請注意,authors是Book 私有字段,AuthorRef實例化在私有方法createAuthorRef中發生。因此聚合之外的任何內容都不能直接訪問它。Spring Data JDBC絕不需要這樣做,但DDD鼓勵這么做。

下面是測試:

@Testpublic void booksAndAuthors() { Author author = new Author(); author.name = "Greg L. Turnquist"; author = authors.save(author); Book book = new Book(); book.title = "Spring Boot"; book.addAuthor(author); books.save(book); books.deleteAll(); assertThat(authors.count()).isEqualTo(1);}

上述完成了我們設想功能:刪除書籍后,并沒有將書籍作者數據表數據全部刪除,雖然作者是書籍的一個私有字段。

總結一下:

Spring Data JDBC不支持多對一或多對多關系。要對這些進行建模,請使用ID。

這鼓勵了領域模型的清晰模塊化。

通過類似的思路,避免雙向依賴。聚合內部的引用是從聚合根到元素。聚合之間的引用使用只在一個關聯方向上使用ID表示。此外,如果需要反向導航,請在存儲庫中使用查詢方法。這樣能清楚確定哪個聚合負責維護引用。

banq注:是不是每個實體都需要一個對應數據表?根據具體情況,Order和OrderItem之間生命周期是一致的,刪除訂單,訂單條目也沒有存在意義;而Book和Author則不是生命周期一致的,Book可能是當前有界上下文的聚合根,而Author是另外一個有界上下文如作者管理系統的聚合根,如果刪除Book同時,也將Author刪除,其實是不符合要求的,這時候應該將Author作為值對象看待,Author的Id就是一個值,然后建立一個類AuthorRef ,包含這個值,作為被Book引用的對象,這樣就不是整個Author實體聚合對象被Book引用了。

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網。


注:相關教程知識閱讀請移步到JAVA教程頻道。
發表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發表
亚洲香蕉成人av网站在线观看_欧美精品成人91久久久久久久_久久久久久久久久久亚洲_热久久视久久精品18亚洲精品_国产精自产拍久久久久久_亚洲色图国产精品_91精品国产网站_中文字幕欧美日韩精品_国产精品久久久久久亚洲调教_国产精品久久一区_性夜试看影院91社区_97在线观看视频国产_68精品久久久久久欧美_欧美精品在线观看_国产精品一区二区久久精品_欧美老女人bb
日韩成人中文电影| 亚洲伊人久久综合| 亚洲已满18点击进入在线看片| 中文字幕久热精品在线视频| 亚洲欧美日韩中文在线制服| 亚洲成人国产精品| 在线性视频日韩欧美| 亚洲风情亚aⅴ在线发布| 欧美三级欧美成人高清www| 最新国产精品亚洲| 午夜精品久久17c| 精品久久久免费| 成人a视频在线观看| 国产精品直播网红| 亚洲色图在线观看| 国产精品久久久久久久久久久久| 亚洲免费人成在线视频观看| 欧美成人亚洲成人日韩成人| 亚洲国产欧美日韩精品| 成人免费在线网址| 国产精欧美一区二区三区| 国产成人久久精品| 中文字幕在线观看日韩| 亚洲欧美精品suv| 亚洲欧美制服中文字幕| 日韩av在线影视| 欧美性受xxx| 欧美壮男野外gaytube| 亚洲国产精品久久精品怡红院| 热草久综合在线| 亚洲国产精品成人av| 青草成人免费视频| 亚洲国产精品视频在线观看| 91国内揄拍国内精品对白| 久久精品国产精品亚洲| 55夜色66夜色国产精品视频| 欧美尺度大的性做爰视频| 黑丝美女久久久| 亚洲人a成www在线影院| 国产精品亚洲激情| 欧美性猛交丰臀xxxxx网站| 日韩中文字幕国产| 欧美成人精品影院| 国产亚洲a∨片在线观看| 国产成人精品在线观看| 久久香蕉国产线看观看网| 欧美日韩午夜激情| 欧美电影免费播放| 一区二区三区四区精品| 日韩久久精品成人| 色综合天天狠天天透天天伊人| 国产精品三级在线| 日韩在线视频一区| 久久免费少妇高潮久久精品99| 国产精品高清免费在线观看| 色偷偷偷亚洲综合网另类| 欧美激情国产高清| 欧美洲成人男女午夜视频| 亚洲精品动漫久久久久| 国产精品久久久久久久久久尿| 青草热久免费精品视频| 欧美在线免费视频| 亚洲一区av在线播放| 成人免费淫片视频软件| 一本大道亚洲视频| 国产精品久久久久久久9999| 中文字幕日韩视频| 97色在线视频| 欧美国产日韩免费| 久久韩国免费视频| 日韩av免费在线看| 亚洲自拍在线观看| 成人av在线天堂| 国产成人精品久久| 亚洲高清免费观看高清完整版| 精品久久久久久久久久| 日韩欧美中文在线| 色视频www在线播放国产成人| 久久久中文字幕| 色综合天天综合网国产成人网| 国产成人一区二区三区| 久久久国产一区二区三区| 日韩在线观看电影| 久久精品国产一区| 亚洲性生活视频在线观看| 久久精品一区中文字幕| 欧美日韩国产成人高清视频| 国产精品日韩欧美大师| 狠狠色噜噜狠狠狠狠97| 欧美夜福利tv在线| 国产精品久久77777| 欧美久久精品午夜青青大伊人| 久久久久久久久久久成人| 中文字幕视频一区二区在线有码| 亚洲自拍小视频免费观看| 欧美成人精品不卡视频在线观看| 成人午夜在线观看| 久久人人爽人人| www.美女亚洲精品| 国产精品永久免费观看| 最近2019年好看中文字幕视频| 久久久久久久97| 日韩视频免费观看| 亚洲护士老师的毛茸茸最新章节| 欧美成人精品一区二区| 欧美国产日韩一区二区| 久久久久久久久国产精品| 国产精品v日韩精品| 国产精品久久久久国产a级| 96精品视频在线| 国产日韩欧美成人| 日韩一区av在线| 欧美日韩国产色| 亚洲国产日韩欧美在线99| 欧美激情在线观看| 国产精品igao视频| 这里只有精品丝袜| 最近中文字幕日韩精品| 久久免费成人精品视频| 一区二区三区无码高清视频| 亚洲精品视频免费| 亚洲二区中文字幕| 国产日韩欧美中文在线播放| 久久久久日韩精品久久久男男| 欧美精品国产精品日韩精品| 亚洲精品一区二区三区婷婷月| 精品日本美女福利在线观看| 欧亚精品中文字幕| 国产成人精品免费久久久久| 国产精品丝袜高跟| 国产精品香蕉国产| 欧美猛交ⅹxxx乱大交视频| 超碰日本道色综合久久综合| 久久精品亚洲精品| 性欧美激情精品| 久久精品色欧美aⅴ一区二区| 精品色蜜蜜精品视频在线观看| 日本高清+成人网在线观看| 久久久精品一区| 国产日韩精品在线观看| 亚洲精品视频免费在线观看| 国产91精品高潮白浆喷水| 国产精品美女www| 久久久精品久久久| 国产视频精品自拍| 欧美日韩中文字幕| 91在线直播亚洲| 亚洲成人久久电影| 欧美日韩国产一区中文午夜| 精品国产成人在线| 日韩成人在线网站| 国产精品一区二区三区在线播放| 日韩经典中文字幕在线观看| 一区二区三区四区视频| 亚洲欧美一区二区三区四区| 精品亚洲一区二区三区| 国产成人亚洲综合| 91精品久久久久久久久久久| 欧美激情一级精品国产| 欧美精品免费在线观看| 一本色道久久综合亚洲精品小说| 亚洲欧美激情四射在线日| 欧美裸体xxxx极品少妇| 成人av资源在线播放|