DDD 中的聚合根持久化问题(使用关系型数据库) - V2EX
V2EX = way to explore
V2EX 是一个关于分享和探索的地方
Sign Up Now
For Existing Member  Sign In
请不要在回答技术问题时复制粘贴 AI 生成的内容
asanelder

DDD 中的聚合根持久化问题(使用关系型数据库)

  •  
  •   asanelder Dec 1, 2020 4689 views
    This topic created in 1975 days ago, the information mentioned may be changed or developed.
    老铁们,好像这种关系型数据库之间表的关系无法完美匹配领域模型啊!

    你们在使用关系型数据库时怎么持久化聚合根的?

    有没有好的思路和实践?
    32 replies    2023-04-19 14:35:40 +08:00
    xuanbg
        1
    xuanbg  
       Dec 1, 2020
    不是你的数据模型设计错了,就是你的领域模型设计错了……
    xiangwan
        2
    xiangwan  
       Dec 1, 2020 via Android
    了解下领域模型和数据模型解耦合
    asanelder
        3
    asanelder  
    OP
       Dec 1, 2020
    @xiangwan #2 有没有好的文章,铁子,找了半天没找到。。。
    @xuanbg #1 感觉都没错啊。。网上说的最好的方式是使用 mongodb 。。。
    xiangwan
        4
    xiangwan  
       Dec 1, 2020 via Android
    领悟模型有人也叫成业务模型,对象模型,可以换着搜搜
    有个解藕思路是,在仓储层转换领域对象和数据对象
    主要是转换下思维,不执意让数据表和领域模型完全对应。可以像上面提到的,在数据访问层进行转换
    asanelder
        5
    asanelder  
    OP
       Dec 1, 2020
    @xiangwan #4 俺用的 spring data jpa,也是想在数据访问层做做文章。

    之前没用过 jpa 和 ddd,俺也在摸索中。。。

    谢谢铁子给的思路,俺慢慢尝试。
    CoderGeek
        6
    CoderGeek  
       Dec 1, 2020
    我最近也在搞 ddd 费劲
    xiangwan
        7
    xiangwan  
       Dec 1, 2020 via Android
    @asanelder 不客气

    @CoderGeek 准备放弃 ddd 了。
    最近觉着 ddd 倾向把问题复杂化,不敏捷。
    imherer
        8
    imherer  
       Dec 1, 2020
    @CoderGeek
    @xiangwan

    搞了 2 个多星期的 DDD,一开始很不习惯,弄下来之后发现还是挺爽的
    CoderGeek
        9
    CoderGeek  
       Dec 2, 2020
    我最近也在搞 ddd 费劲
    @xiangwan
    真的需要业务专家 不是说技术的问题,
    起码第一步不能画跑偏了
    xiangwan
        10
    xiangwan  
       Dec 2, 2020 via Android
    @CoderGeek ddd 有很多优点,但是在技术层面,“原教旨“ddd 用的方式很复杂,例如:事件溯源,命令查询分离;
    “聚合根“也是一个好坏参半的解决方案,除了本贴持久化的问题,还面临选择是用充血模型还是贫血模型,大部分贫血模型的项目其实都是伪 ddd 。

    @imherer 我尝试好多次 ddd 了,总感觉不够 ddd 。它的“战略工具”确实可以,前阵跟着微服务有火了一把
    imherer
        11
    imherer  
       Dec 2, 2020
    @xiangwan 战略工具 具体指什么 有相关文章或者 demo 吗?
    huifer
        12
    huifer  
       Dec 2, 2020
    聚合根是否需要落地是个问题. 有些可能是联合查询对外暴露, 聚合根落地的话简单的形式就是各个外键 id 存储
    asanelder
        13
    asanelder  
    OP
       Dec 2, 2020
    @xiangwan #10 确实,完全遵循原教旨很麻烦,但俺也是从中借鉴一些思想,比如领域模型,充血对象等等。

    之所以想实践 DDD,就是感觉之前做的所有 Web 无关是 CRUD,实在是受不鸟啦!
    @huifer #12 俺在使用 spring data jpa 做持久化,持续摸索中。。。
    CoderGeek
        14
    CoderGeek  
       Dec 2, 2020
    @xiangwan 我的想法是没有所谓绝对标准,不管什么架构模式都是一步步沉淀的 还是得明白业务领域,不说写什么的问题,起码前期梳理的全面 大模块不会跑偏, 这个东西确实需要有一定沉淀的业务人员和技术人员 要么连好事都算不上 浪费时间
    CoderGeek
        15
    CoderGeek  
       Dec 2, 2020
    @imherer
    @xiangwan
    取其精华,去其糟粕 ;有好有坏
    leoskey
        16
    leoskey  
       Dec 2, 2020
    我是参考 https://www.cnblogs.com/uoyo/p/12167224.html 的实践,有些落地难得地方需要妥协
    iamppz
        17
    iamppz  
       Dec 2, 2020
    聚合根和数据库对象之间通过 Converter 来 serialize 和 deserialize
    asanelder
        18
    asanelder  
    OP
       Dec 2, 2020
    @leoskey #16 感谢,俺看看

    @iamppz #17 确实是一个方法,但俺想利用 ORM 直接将数据库对象映射到聚合根,多做一次转换好麻烦啊
    asanelder
        19
    asanelder  
    OP
       Dec 2, 2020
    @xiangwan #4
    @iamppz #17

    其实这种从领域对象到数据对象的转换就是因为关系型数据库中的数据模型无法完全匹配领域模型,而俺看 mongodb 的文档可以完全匹配,那么,是不是换成 mongodb 就没这种烦恼了呀
    hantsy
        20
    hantsy  
       Dec 2, 2020
    技术上使用 Spring Data 的话,它已经抽象了 DomainEvent,AggregrateRoot 处理,适用大多数 Spring Data 子项目,包括 Spring Data JPA, 请参考 Spring Data 文档。
    https://docs.spring.io/spring-data/data-commons/docs/current/api/org/springframework/data/domain/DomainEvents.html
    https://docs.spring.io/spring-data/data-commons/docs/current/api/org/springframework/data/domain/AbstractAggregateRoot.html


    另外代码层面实现上,充分利用 Spring 特性,比如 ApplicationEvent 处理 Domain Event,Message Broker 来实现跨 Domain (不同的 BoundedContext )数据同步。

    Domain 建模,Aggregate Root 的颗粒度源于实践,怎么具体处理,取决你的长期经验。经验需要不断积累,不断试错和踩坑,技术架构也是需要不断演进的。理论上很多东西一看就明白,落地很难,比如 Single Reponsibility 拿到国内,细化接口规范后,有些人就只会比较字符数量,便得出结论是增加了任务量。
    hantsy
        22
    hantsy  
       Dec 2, 2020
    kvkboy
        23
    kvkboy  
       Dec 2, 2020
    kvkboy
        24
    kvkboy  
       Dec 2, 2020
    @kvkboy 手快不小心不小心发出去了,可以看看这个文章,挺有参考的价值。不过更多的还是需要看书,ddd 经典的两本一本理论一本落地
    hantsy
        25
    hantsy  
       Dec 2, 2020
    @kvkboy 六边形和洋葱不错,特别六边形,把系统进出接口进行隔离,感观比较清晰。

    文章提到的 Cargo 例子作为 Eric 原书 DDD 的一部分,使用了 Spring 框架,现在已经有各种版本了。

    Eclipse EE4J 官方也维护了一个基于 Java EE/Jakarta EE 规范的 Cargotraker.

    https://github.com/eclipse-ee4j/cargotracker
    hantsy
        26
    hantsy  
       Dec 2, 2020
    asanelder
        27
    asanelder  
    OP
       Dec 2, 2020
    @hantsy #20
    @kvkboy #23

    感谢二位铁子,文章俺都看了,也有一些思路了,正在实践中。。。
    Kirsk
        28
    Kirsk  
       Dec 2, 2020 via Android
    Ddd 是业务建模 不是数据建模 不懂业务还是不要玩
    iamppz
        29
    iamppz  
       Dec 2, 2020
    @asanelder 个人感觉跟数据库没有关系,你需要一个合适的 ORM,我用的 Spring Data JPA
    hantsy
        30
    hantsy  
       Dec 2, 2020
    @iamppz 理想的设计层面,Domain 建模不会考虑到数据库,到真正实现的时候才考虑。如果 Domain 模型与数据库有关系,就是 DDD 中的 Entity,最终要持久化,不得不考虑到数据库的问题。

    国内的实践,直到现在很多人做项目或者产品还是本末倒置,需求一到,数据库 Schema 优先设计,使用一些工具自动生成 Entities (或者什么框架需要的类似的东西,这只是数据库的 Entities,与 Domain Model 中 Entities 无关)。从一开始就是面向数据库的,最终业务都是围绕数据存储。以前项目中经常遇到有谈业务的时候,没两句直接会想到数据库存放,页面一个搜索框,能够联想到联合查询等,完全脱离不了数据库思维。
    asanelder
        31
    asanelder  
    OP
       Dec 2, 2020
    @iamppz #29

    俺就是用的 spring data jpa,但是也只是做 orm, m 和 domain object 还是要人工转化一下。

    @hantsy #30

    今天搞定 domain object 和 data object 的转化后,用充血模型写业务,有点爽啊~~~
    dandankele
        32
    dandankele  
       Apr 19, 2023
    @asanelder 用 DDD 充血模型确实爽。。但我还卡在怎么用 jpa 将 data object 与 domain object 进行转化。。目前打算用的是 mapStruct 进行转化。。
    另外还有个问题,spring data jpa 中的 repository 只是用来查询 data object 了吧?我觉得它的 repository 与 DDD 的 repository 其实不是同一个概念了。。
    About     Help     Advertise     Blog     API     FAQ     Solana     932 Online   Highest 6679       Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 52ms UTC 21:21 PVG 05:21 LAX 14:21 JFK 17:21
    Do have faith in what you're doing.
    ubao msn snddm index pchome yahoo rakuten mypaper meadowduck bidyahoo youbao zxmzxm asda bnvcg cvbfg dfscv mmhjk xxddc yybgb zznbn ccubao uaitu acv GXCV ET GDG YH FG BCVB FJFH CBRE CBC GDG ET54 WRWR RWER WREW WRWER RWER SDG EW SF DSFSF fbbs ubao fhd dfg ewr dg df ewwr ewwr et ruyut utut dfg fgd gdfgt etg dfgt dfgd ert4 gd fgg wr 235 wer3 we vsdf sdf gdf ert xcv sdf rwer hfd dfg cvb rwf afb dfh jgh bmn lgh rty gfds cxv xcv xcs vdas fdf fgd cv sdf tert sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf sdf shasha9178 shasha9178 shasha9178 shasha9178 shasha9178 liflif2 liflif2 liflif2 liflif2 liflif2 liblib3 liblib3 liblib3 liblib3 liblib3 zhazha444 zhazha444 zhazha444 zhazha444 zhazha444 dende5 dende denden denden2 denden21 fenfen9 fenf619 fen619 fenfe9 fe619 sdf sdf sdf sdf sdf zhazh90 zhazh0 zhaa50 zha90 zh590 zho zhoz zhozh zhozho zhozho2 lislis lls95 lili95 lils5 liss9 sdf0ty987 sdft876 sdft9876 sdf09876 sd0t9876 sdf0ty98 sdf0976 sdf0ty986 sdf0ty96 sdf0t76 sdf0876 df0ty98 sf0t876 sd0ty76 sdy76 sdf76 sdf0t76 sdf0ty9 sdf0ty98 sdf0ty987 sdf0ty98 sdf6676 sdf876 sd876 sd876 sdf6 sdf6 sdf9876 sdf0t sdf06 sdf0ty9776 sdf0ty9776 sdf0ty76 sdf8876 sdf0t sd6 sdf06 s688876 sd688 sdf86