当前位置: 首页 > news >正文

通俗地讲述DDD的设计

通俗地讲述DDD的设计

  • 前言
  • 为什么要使用DDD
  • DDD架构分层重构实践
    • 关键问题解决方案
      • 通过​​领域事件机制​​解耦服务依赖:
      • 防止逻辑下沉
    • 领域划分
      • 电商场景下的领域划分
  • 结语
  • 完结撒花,如有需要收藏的看官,顺便也用发财的小手点点赞哈,如有错漏,也欢迎各位在评论区评论!

前言

平时我们在谈论到DDD(驱动领域设计)的时候,往往感觉讳莫如深,特别是市面上对于它的定义、落地实现策略这些,都没有明确的解释,似乎一百个人有一百种解释,以至于我们在设计架构层级以及具体代码实现时,都不会用到DDD的设计思想,于是本文会按笔者理解DDD的概念以及设计原则去通俗描述,尽量让诸位培养成下意识使用它的习惯;

为什么要使用DDD

在传统分层架构(API层-业务层-Service层-DAO层)实践中,我们常遇到以下典型问题:

​​Service层相互调用导致的链式灾难​​(如A->B->C->D的深度调用)
​​Common模块过度抽象引发的架构退化​​(业务层公共代码下沉导致Service层失去业务语义)
​​调用关系失控​​(业务调用可能意外穿透到DAO层,层级边界模糊)
这些问题的本质是​​业务逻辑与技术实现耦合度过高​​,而领域驱动设计(Domain-Driven Design)为解决这些问题提供了新的视角。

DDD架构分层重构实践

// 示例包结构
com.example
├── api        // API层:接口定义、DTO转换
├── application // 业务层:业务流程编排
├── domain     // 领域层:核心业务逻辑
└── infra      // 基础设施层:数据库/外部服务访问

API层​​:定义对外契约,完成DTO与领域对象的双向转换。建议采用门面模式隔离外部参数与内部模型。
​​业务层​​:编排领域服务,处理跨领域协作。此处应保持无状态,仅包含流程控制逻辑。
​​领域层​​:封装业务规则(对于某一功能具体的代码实现),通过聚合根维护领域完整性。建议每个领域模块独立成包(如order/payment)。
​​DAO层​​:重构为基础设施层,实现仓储接口,支持多数据源适配。
注意:ddd的事务管理应该放在业务层,还是领域层,各有各的风险 ,业务层因为是编排调用多个服务,就会有长事务的问题,而在领域层则是有一致性问题;(因为如果把事物放在application层,增删改之间掺杂比较耗时的其他操作就会造成大事物。放在domain层,其他领域可能会有关联)

关键问题解决方案

通过​​领域事件机制​​解耦服务依赖:

// 订单创建后发布领域事件
public class OrderService {@Transactionalpublic void createOrder(Order order) {orderRepository.save(order);eventPublisher.publish(new OrderCreatedEvent(order.getId()));}
}// 库存服务监听事件
@Component
public class InventoryHandler {@EventListenerpublic void handle(OrderCreatedEvent event) {inventoryService.lockStock(event.getOrderId());}
}

防止逻辑下沉

建立​​分层防护机制​​:

在领域层定义仓储接口
基础设施层实现具体仓储
通过依赖注入控制反转

// 领域层定义接口
public interface OrderRepository {Order findById(OrderId id);
}// 基础设施层实现
@Repository
public class JpaOrderRepository implements OrderRepository {// 具体实现
}

领域划分

ddd的精华就是在于领域划分,而领域模型的设计目的
1、业务和代码要对齐,不至于一个简单的业务,要用成堆的代码解决,还要回想整个代码流程才明白这个业务;
2、让测试、开发、产品对某一个bug的描述要一致;
3、减少修改难度和影响面;
说明:
聚合根:就是bo、vo、do这些不同层有自己的返回对象类;
领域调用其他领的中间会有一个防腐层(防腐层是指对参数的转换,把这个参数转换抽取出来,传递的就是聚合根,有了防腐层是为了防止腐化其他领域的代码)

电商场景下的领域划分

┌──────────────┐ ┌─────────────┐
│ 订单域 │◄─────►│ 支付域 │
│ - 订单创建 │ │ - 支付处理 │
│ - 订单状态机 │ └─────────────┘
└───────┬──────┘

┌──────────────┐
│ 库存域 │
│ - 库存扣减 │
│ - 库存预警 │
└──────────────┘
不同层级之间传递参数,要注意转换和收敛,就是上层调用多个下层,也不要直接一个参数对象从头走到尾,而是将对象所需属性转换成需要相应的对象;

结语

DDD的真正价值在于​​建立业务与技术的统一语言​​。建议采用渐进式改造:

从核心子域开始试点
建立领域字典统一术语
通过持续重构优化模型
技术架构的演进永无止境,但通过DDD建立的清晰领域边界,能使系统在业务快速变化中保持足够的弹性。

完结撒花,如有需要收藏的看官,顺便也用发财的小手点点赞哈,如有错漏,也欢迎各位在评论区评论!

相关文章:

通俗地讲述DDD的设计

通俗地讲述DDD的设计 前言为什么要使用DDDDDD架构分层重构实践关键问题解决方案通过​​领域事件机制​​解耦服务依赖:防止逻辑下沉 领域划分电商场景下的领域划分 结语完结撒花,如有需要收藏的看官,顺便也用发财的小手点点赞哈,…...

【学Rust写CAD】34 精确 Alpha 混合函数(argb.rs补充方法)

源码 #[inline]pub fn over_exact(self, dst: Argb) -> Argb {let a 255 - self.alpha32();let t dst.rb() * a 0x80_00_80;let mut rb (t ((t >> 8) & Argb::MASK)) >> 8;rb & Argb::MASK;rb self.rb();// saturaterb | 0x1000100 - ((rb >&…...

10种电阻综合对比——《器件手册--电阻》

二、电阻 前言 10种电阻对比数据表 电阻类型 原理 特点 应用 贴片电阻 贴片电阻是表面贴装元件,通过将电阻体直接贴在电路板上实现电路连接 体积小、重量轻,适合高密度电路板;精度高、稳定性好,便于自动化生产 广泛应用于…...

SpringCloud入门及创建分布式项目

1、了解微服务 1.1 什么是微服务 微服务是一种架构风格一个应用拆分为一组小型服务每个服务运行在自己的进程内,也就是可独立部署和升级服务之间使用轻量级HTTP交互服务围绕业务功能拆分可以由全自动部署机制独立部署去中心化,服务自治。服务可以使用不同…...

xv6启动过程

entry,S -> start.c -> main.c -> proc.c中的 userinit 函数 -> initcode.S -> init.c entry.S // entry.S# qemu -kernel loads the kernel at 0x80000000# and causes each CPU to jump there.# kernel.ld causes the following code to# be placed at 0x800…...

【秣厉科技】LabVIEW工具包——OpenCV 教程(18):highgui 模块

文章目录 前言highgui 模块总结 前言 需要下载安装OpenCV工具包的朋友,请前往 此处 ;系统要求:Windows系统,LabVIEW>2018,兼容32位和64位。 highgui 模块 尽量别用,要不我删了吧? LabVIEW…...

基于OPENCV的图像透视矫正

这段代码的主要功能是对输入的图像进行透视矫正。它会读取一张图像,检测图像中最大的四边形轮廓,然后对该四边形区域进行透视变换,将其矫正为正视图,最后保存矫正后的图像。 模块导入说明 python import cv2 import numpy as n…...

数据结构----------顺序查找,折半查找和分块查找(java实现)

import java.util.Arrays;//顺序查找法 public class Main {public static void main(String[] args) {//查找表int[] arr {4, 3, 5, 1, 2};System.out.print("5在数组中的索引:");System.out.println(SearchSeq(arr, 5));Arrays.sort(arr);System.out.print("…...

整数编码 - 华为OD统一考试(A卷、JavaScript)

题目描述 实现一种整数编码方法,使得待编码的数字越小,编码后所占用的字节数越小。 编码规则如下: 编码时7位一组,每个字节的低7位用于存储待编码数字的补码。字节的最高位表示后续是否还有字节,置1表示后面还有更多的字节&…...

CompletableFuture:整合、超时、完成事件与批量处理

引言 在异步编程实践中,我们不仅需要处理单个任务的执行流程,更需要应对多个异步任务之间的复杂交互。本文将通过实际案例解析以下核心功能: 双任务整合:合并两个独立任务的结果高效超时控制:防止异步操作无限等待完…...

【LeetCode 热题100】45:跳跃游戏 II(详细解析)(Go语言版)

🚀 力扣 45:跳跃游戏 II(全解法详解) 📌 题目描述 给你一个非负整数数组 nums,表示你最初位于数组的第一个位置。 数组中的每个元素表示你在该位置可以跳跃的最大长度。 你的目标是使用 最少的跳跃次数 到…...

【C/C++】滑动谜题(leetcode T773)

核心考点:广度优先搜索 (BFS)、哈希表、字符串、状态转移 题目描述: 在一个 2 x 3 的板上(board)有 5 块砖瓦,用数字 1~5 来表示, 以及一块空缺用 0 来表示。一次 移动 定义为选择 0 与一个相邻的数字(上…...

python用x08覆盖上一次输出来模拟控制台等待效果,pycharm运行sys.stdout.write在控制台无打印的解决方法

一个多进程程序,主进程阻塞,子进程不断打印等待效果直到主进程结束,原理是\x08在ascii中表示退格键,理解为打印完后立马删掉打印下一个内容。 import sys, time import multiprocessing DELAY 0.1 DISPLAY [ |, /, -, \\ …...

【嵌入式开发】使用Linux系统调用编程练习

一、进程和线程的概念及基础用法 在Linux系统中,进程(Process)和线程(Thread)是操作系统进行任务调度的基本单位,它们既有联系又有区别。 1.1 进程和线程介绍 1.1.1 进程(Process&#xff09…...

React框架的Concurrent Mode

以下是关于 Concurrent Mode 的系统梳理: 一、Concurrent Mode 的诞生背景 传统渲染的局限性 同步阻塞:React 15 的 Stack Reconciler 无法中断渲染流程优先级缺失:用户交互与后台任务同等对待资源竞争:网络请求与渲染任务无法智能调度核心设计目标 可中断渲染:允许高优先…...

ER-图,详情和画法

一、E-R图的核心元素 1.实体 表示现实中对象或概念,用矩形表示 示例:用户、老师、学生 2.属性 描述实体的特征,用椭圆表示。 分为主键(用户id) 和非主键(用户昵称) 3.关系 表示实体间的…...

深度学习图像分类数据集—十种西红柿病态叶识别分类

该数据集为图像分类数据集,适用于ResNet、VGG等卷积神经网络,SENet、CBAM等注意力机制相关算法,Vision Transformer等Transformer相关算法。 数据集信息介绍:10种西红柿病态叶识别分类:Bacterial_spot,Earl…...

【Flask开发】嘿马文学web完整flask项目第3篇:2.用户认证,2.用户认证【附代码文档】

教程总体简介:2. 目标 1.1产品与开发 1.2环境配置 1.3 运行方式 1.4目录说明 1.5数据库设计 2.用户认证 Json Web Token(JWT) 3.书架 4.1分类列表 5.搜索 5.3搜索-精准&高匹配&推荐 6.小说 6.4推荐-同类热门推荐 7.浏览记录 8.1配置-阅读偏好 8.配置 9.1项目…...

基于Pyhon的京东笔记本电脑数据可视化分析系统

【Python】基于Pyhon的京东笔记本电脑数据可视化分析系统 (完整系统源码开发笔记详细部署教程)✅ 目录 一、项目简介二、项目界面展示三、项目视频展示 一、项目简介 本项目基于Python语言开发,通过Flask框架与Bootstrap的结合,实…...

『不废话』之Llama 4实测小报

2025年4月5日Llama 4一开源,随后OpenRouter等平台就提供免费调用。对于中文社区来,官方的测评结果其实意义不大(原因先按下不表),就看知乎、微博、B站、twitter上的真实感受,最重要的是自己的真实案例测评。…...

llama.cpp 和 vLLM 的详细对比分析

llama.cpp 和 vLLM 的详细对比分析,基于最新技术动态(2025年4月)整理: 1. 核心定位 维度llama.cppvLLM设计目标轻量化边缘计算,突破硬件限制(如手机/树莓派)企业级高性能推理,优化G…...

Windows 操作系统使用vscode 配置GTK4

本篇教程,主要介绍在vscode中如何配置使用GTK4,并运行一个简易的入门案例。 一、程序代码 1、demo.cpp #include <gtk/gtk.h>// 定义一个回调函数,用于处理按钮点击事件 static void on_button_clicked(GtkButton *button...

swift-汇编分析多态原理、init

一、结构体和类的本质区别 结构体 编译完就知道调用谁 类 &#xff08;类似c 的虚函数表&#xff09; 12 直接将对象将来要调用的函数内存地址提前放进类型信息里面&#xff0c;这些类型信息编译完就确定你将来要调用谁&#xff0c;运行过程过程中就去那块内存里面找 方法的存…...

Docker基础2

如需转载&#xff0c;标记出处 本次我们将下载一个 Docker 镜像&#xff0c;从镜像中启动容器 上一章&#xff0c;安装 Docker 时&#xff0c;获得两个主要组件&#xff1a; Docker 客户端 Docker 守护进程&#xff08;有时称为“服务器”或“引擎”&#xff09; 守护进程实…...

labelme json 标签转yolo txt【记录】

01 labelme json 转 txt&#xff08;w_convert_labelme_to_yolo.py&#xff09; #WT 将labelme json标签格式转换为YOLO txt格式 # 导入所需模块 import cv2 # OpenCV用于图像处理 import os # 操作系统路径管理 import json # JSON文件解析 import glob # 文件通配符搜索…...

Java 集合框架与 Stream 流深入剖析(重点详细讲解)

目录 引言 一、ArrayList 1. 概述 2. 特点 动态扩容 初始容量 扩容倍数 随机访问高效 插入和删除效率低 3. 代码示例 4. 分析 二、HashSet 1. 概述 2. 特点 唯一性 插入、删除和查找效率高 无序性 3. 代码示例 4. 分析 三、HashMap 1. 概述 2. 特点 键唯…...

实操(多线程特点、健壮性降低、缺乏访问控制)Linux

线程 创建两个线程 makefile&#xff08;添加原生线程库&#xff09; mythread:thread.ccg -o $ $^ -stdc11 -lpthread .PHONY:clean clean:rm -f mythreadthread.cc #include <iostream> #include <pthread.h> #include<unistd.h>using namespace std;…...

微信小程序学习实录12:掌握大数据量轨迹展示的MySQL结构设计

获取经纬度信息后&#xff0c;mysql建立数据表po_trajectory&#xff0c;字段包含tra_id、longitude、latitude、tra_time和openid。 为微信小程序创建的 po_trajectory 数据表&#xff0c;字段包含 tra_id、longitude、latitude、tra_time 和 openid&#xff0c;从结构设计上…...

语法: result=ldexp (value, exp);

LDEXP( ) 语法: resultldexp (value, exp); 参数: value是一个浮点数; exp是一个有符号的整型数; 返回值: result同value保持一致,是一个浮点数,结果是value乘以2的exp次方. 功能: ldexp( ) 该函数是用一个浮点数乘以2的多少(整数)次方. 有效性: 适合所有的CPU设备…...

STM32学习之硬件FPU(原理篇)

&#x1f4e2;&#xff1a;如果你也对机器人、人工智能感兴趣&#xff0c;看来我们志同道合✨ &#x1f4e2;&#xff1a;不妨浏览一下我的博客主页【https://blog.csdn.net/weixin_51244852】 &#x1f4e2;&#xff1a;文章若有幸对你有帮助&#xff0c;可点赞 &#x1f44d;…...

QEMU源码全解析 —— 块设备虚拟化(15)

接前一篇文章:QEMU源码全解析 —— 块设备虚拟化(14) 本文内容参考: 《趣谈Linux操作系统》 —— 刘超,极客时间 《QEMU/KVM源码解析与应用》 —— 李强,机械工业出版社 特此致谢! QEMU初始化阶段的块设备虚拟化 本回解析virtio_blk_device_realize函数中的virtio_a…...

Web3(阶段一:入门)——哈希算法

一、简述 哈希算法&#xff08;Hash Algorithm&#xff09;是一种将任意长度的输入数据转换为固定长度输出&#xff08;哈希值&#xff09;的数学函数。其核心作用是通过不可逆的计算生成唯一标识数据的 “数字指纹”&#xff0c;广泛应用于数据完整性验证、密码学、区块链等领…...

高频面试题(含笔试高频算法整理)基本总结回顾63

干货分享&#xff0c;感谢您的阅读&#xff01; &#xff08;暂存篇---后续会删除&#xff0c;完整版和持续更新见高频面试题基本总结回顾&#xff08;含笔试高频算法整理&#xff09;&#xff09; 备注&#xff1a;引用请标注出处&#xff0c;同时存在的问题请在相关博客留言…...

如何深入理解C#中的备忘录模式(Memento Pattern)设计模式

在软件开发中&#xff0c;设计模式是一种解决特定问题的通用方法&#xff0c;而备忘录模式&#xff08;Memento Pattern&#xff09;是其中一种用于保存对象状态的结构型设计模式。它允许你在不暴露对象内部结构的情况下&#xff0c;保存和恢复对象的状态。本文将深入探讨C#中的…...

存储引擎 / 事务 / 索引

1. 存储引擎 MySQL 中特有的术语。 &#xff08;Oracle 有&#xff0c;但不叫这个名字&#xff09; 是一种表存储 / 组织数据的方式 不同的存储引擎&#xff0c;表存储数据的方式不同 1.1 查看存储引擎 命令&#xff1a; show engines \g&#xff08;或大写&#xff1a;G…...

药店管理系统

https://download.csdn.net/download/weixin_57836618/90572873 软件架构 Java SpringBoot Mybatis/Mybatis-plus Mysql 项目功能说明 促销管理&#xff1a;零售出库、零售退货采购管理&#xff1a;采购订单、采购入库、采购退货销售管理&#xff1a;销售订单、物流信息、…...

Kafka 的发展历程

Kafka 作为一个高性能的分布式消息流平台&#xff0c;从诞生到现在已经有了长足的发展&#xff0c;经历了多个版本的迭代。下面是 Kafka 的 发展历史、版本迭代 以及 新特性 的概述。 1. Kafka 的诞生与早期发展 2010年&#xff1a;Kafka 由 LinkedIn 的工程师 Jay Kreps、Ne…...

PowerBI 之DAX 3:文本、信息、财务、时间智能函数

文章目录 一、文本函数1.1 FORMAT函数1.1.1 数字格式1.1.2 日期/时间格式1.1.3 自定义格式 1.2 CONCATENATE与CONCATENATEX1.2.1 返回多个类别名称1.2.2 返回多个类别的名称和数据&#xff0c;并排序 1.3 使用SEARCH进行模糊查找 二、信息函数2.1 ISINSCOPE 三、财务函数3.1 PV…...

GESP C++三级 知识点讲解

C编程三级标准 (一)知识点详述 (1)了解二进制数据编码:原码、反码、补码。 (2)掌握数据的进制转换:二进制、八进制、十进制、十六进制。 (3)掌握位运算:与(&)、或(|)、非(~)、异或(^)、左移(<<)、右移(>>)的基本使用方法及原理。 (4)了解算法的概念与描述&…...

如何访问和使用Sora:OpenAI视频生成模型的完整指南

OpenAI的Sora作为革命性的视频生成模型&#xff0c;能够根据文本提示创建长达60秒的高质量视频内容。本教程将详细介绍目前Sora的使用方法和访问途径。 一、Sora当前访问状态&#xff08;2024年3月更新&#xff09; 目前Sora仍处于有限访问阶段&#xff0c;OpenAI采取了分阶段…...

MyBatis 分页插件使用教程

MyBatis 分页插件使用教程 MyBatis 是一款优秀的持久层框架&#xff0c;但原生的 MyBatis 并不支持分页查询。为了简化分页操作&#xff0c;MyBatis 官方和第三方提供了多种分页插件&#xff0c;最常用的就是 MyBatis-Plus 的分页插件。本文详细介绍 MyBatis-Plus 分页插件的使…...

OpenDriveVLA:通过大型视觉-语言-动作模型实现端到端自动驾驶

25年3月来自慕尼黑工大和慕尼黑大学的论文“OpenDriveVLA: Towards End-to-end Autonomous Driving with Large Vision Language Action Model”。 OpenDriveVLA&#xff0c;一种专为端到端自动驾驶而设计的视觉-语言-动作 (VLA) 模型。OpenDriveVLA 以开源预训练大型视觉-语言…...

深入探究C++ 运算符重载:以日期类为例

目录 前言 一、运算符重载基础 1.1 运算符重载原理 1.2 示例代码 二、赋值运算符重载 2.1 赋值运算符重载格式 2.2 代码实现 2.3 注意事项 三、前置和后置重载 3.1 前置重载 3.2 后置重载 四、日期类的完整实现 4.1 获取某月天数 4.2 完整类定义 五、总结 前言 …...

2024第十五届蓝桥杯大赛软件赛省赛Java大学B组 报数游戏 类斐波那契循环数 分布式队列 食堂 最优分组 星际旅行 LITS游戏 拼十字

目录 A 报数游戏 B 类斐波那契循环数 C 分布式队列 D 食堂 E 最优分组 F 星际旅行 G LITS 游戏 H 拼十字 今天心血来潮把去年的题目又做了一遍... 本人去年大一 拿的是全省第五进的国赛 而如今的已经是一名 codeforces 1500 分的入门级别的算竞选手了 下周又是蓝桥杯…...

4月6日随笔

一觉起来十点多 其实六点和九点分别醒过一次。 起来之后点了个侍卫草推荐的猪排饭&#xff0c;真的巨好吃&#xff0c;猪排很脆&#xff0c;溏心蛋也很香 但是因为酒店十二点半要退房&#xff0c;就匆匆吃完了猪排和一半米饭就走了 今天下午在科技楼写了一会作业&#xff0c…...

[GN] sigrokdecode 模块学习指南 --- 准备阶段

系列文章目录 文章目录 系列文章目录前言指南libsigrokdecode 学习一、构建环境安装libsigrokdecode安装 sigrok-cli&#xff08;命令行工具&#xff09;安装 PulseView&#xff08;图形界面&#xff09;关联 libsigrokdecode完整验证参数解释 二、BUG解决1. 确保编译时启用了 …...

【力扣hot100题】(056)电话号码的字母组合

依旧是很经典的回溯。 记得当初做这题想了半天电话号码怎么存储&#xff0c;用哈希表就可以解决。 class Solution { public:vector<string> result;string digits;int loc0;unordered_map<char,string> dictionary{{2,"abc"},{3,"def"},{4,…...

kotlin,数字滚动选择

用国内的通义灵码和codegeex都没有弄出来&#xff0c;最后只得用墙外的chatgpt才弄出一个满意的。kotlin真的有点难&#xff0c;好在有AI&#xff0c;让学习没这难了。 package com.example.mynumsetimport android.os.Bundle import androidx.activity.ComponentActivity imp…...

Flask学习笔记 - 数据库

Flask 数据库操作 Flask 提供了多种方式来与数据库进行交互&#xff0c;包括直接使用 SQL 和利用 ORM&#xff08;对象关系映射&#xff09;工具&#xff0c;如 SQLAlchemy。 使用SQLAlchemy创建和管理数据库&#xff1a;使用 db.create_all() 创建表。CRUD 操作&#xff1a;…...

学透Spring Boot — 015. 自废武功——关闭自动配置

这是我的《学透Spring Boot》专栏的第15篇文章&#xff0c;了解更多请移步我的专栏&#xff1a;CSDN Postnull的专栏《学透Spring Boot》 目录 遇到的问题 分析日志 自动配置的过程 解决报错 方法1&#xff1a;添加数据库配置 方法2&#xff1a;关闭JPA自动配置 总结 遇…...