聊聊 ASP.NET 6 整洁架构开发模板
大家好,我是Edison。
最近看了一些整洁架构(CleanArchitecture)的文章,自己和同事也简单写了一个基于整洁架构的ASP.NET 6开发模板在玩。这里就仅仅抛个砖,案例主要以自己根据小组实际情况做了一些裁剪,可能不具有通用的应用性,大家看看就好。
整洁架构的产生背景
微服务架构让DDD(领域驱动设计)焕发了第二春,在DDD的推动下,DDD的分层架构被逐渐推上了舞台。DDD的分层架构就有好多种,例如整洁架构、CQRS和六边形架构等等,每种架构模式虽然提出的时代和背景不同,但其核心理念都是为了设计出“高内聚低耦合”的架构,从而能够实现架构的演进。
DDD分层架构
在欧创新老师的《DDD实战课》中,给出了一个优化后的DDD四层架构,我们可以从下面这张图中看到,从上到下分别是:用户接口层、应用层、领域层和基础层。
与传统的三层架构不同,DDD四层架构的重点在于引入了一个领域层。
领域层的作用是实现企业核心业务逻辑,通过各种校验手段保证业务的正确性。领域层主要体现领域模型的业务能力,它用来表达业务概念、业务状态和业务规则。领域层包含:聚合根、实体、值对象、领域服务等领域模型中的领域对象。对于领域层,领域模型的业务逻辑主要由实体和领域服务来实现。对于实体,一般建议采用充血模型来实现所有与之相关的业务功能。对于领域服务,一般当单个实体不能实现某些功能时,领域服务才会出马,组合聚合内的多个实体来实现复杂的业务逻辑。下图中展示了传统的三层架构与DDD四层架构的对应关系:
整洁架构简单介绍
简而言之,整洁架构是组织软件体系结构的原则,可以轻松面对未来的不确定性,方便代码的重构。同时,它可以帮助我们为特定的领域模型构建服务,从而为将来可能的微服务体系结构做好准备。在Jason Taylor的这篇文章中《Clean Architecture with .NET Core: Gettting Started》中给出了一张经典的图:
在整洁架构中,所有依赖关系都向内流动,而核心层不依赖于其他任何层。基础设施层和展示层依赖于核心层,而不是彼此依赖。在Jason Taylor给出的图中,只有三个圆圈,但在实际中,你可能需要更多,但是你可以以此作为起点,只需要记住让所有依赖都指向内部即可。这里的内部,我们可以将其称为ApplicationCore,也就是Application + Domain。
整洁架构模板搭建
这里我试着搭建了一个基于ASP.NET 6的开发模板,展示层有两种可选:ASP.NET WebAPI / Blazor。需要说明是:该模板仅仅是结合我司实际情况的构想,没有遵循DDD的一些原则思想(DDD是个好东西,但不是所有项目都适用,也不是所有团队都可以用好),也不具有广泛应用性,各位看官看看一笑而过就好。
在我司(一家制造业工厂的IT部),基于我们组的实际人员情况中(开发基础能力较弱,以前的工作基本以运维为主,很少做开发工作)和开发项目的综合复杂度(严格来说,复杂度并不高,以后台管理信息系统为主),我不想引入太多DDD的概念增加学习成本,因此ValueObject和Domain Service被我移除了,但是充血模型的Entity是我们所倡导的,因此,最终的结构如下图所示:
对于展示层,分别使用WebAPI和Blazor实现API和UI的宿主;
对于核心层(ApplicationCore),包含 Application 和 Domain 两个.NET 6.0类库项目。
(1)Application定义了Services、Handlers(对于MQ的Consumers)、Validators(基于FluentValidation的Validators)以及 各种Models(DTO、VO,以及基于AutoMapper的MappingProfiles)。
(2)Domain则定义了实体、枚举、异常、常量等。这一层无需引入过多概念,只需要在原有实体的基础上,使用充血模型,让实体的行为丰富起来即可,这也可以让开发人员很快适应和模仿。
对于基础设施层,也是一个.NET 6.0类库,主要包含了基于EF Core的上下文(DbContext)、实体映射关系(EntityConfiguration)、Repositories、Gateways(针对依赖的外部接口HttpClient实现,可以用HttpClientFactory来实现,也可以用WebApiClient之类的封装项目)、Cache(比如RedisClient的注册)、MessageQueue(比如KafkaClient的注册,取决于你们组用了什么MQ)等。
除了上面的四层之外,设计一个CrossCuting的Shared类库,用于存放一些各个层都可以复用(引用)的帮助类、扩展方法、基类等,用于减少重复。整个项目在Visual Studio中的解决方案目录如下图所示:
整个项目在Visual Studio中的解决方案目录如下图所示:
最终的依赖关系如下:
(1)Domain类库只引用Shared类库(即CrossCutting)。
(2)Application类库引用:Domain、Infrastructure、Shared
这里我们没有让Application不依赖于Infrastructure,是因为我们的DB技术栈已经固定,而且大家也比较习惯于在Infrastructure里边定义Repository、Gateway等的interface,而不是在Application里面定义这些interface,Infrastructure通过应用Application来做实现。
(3)Infrastructure类库引用:Domain、Shared
(4)Web项目引用:Application、Shared(其实这里Application引用了Shared,Web项目无需再添加引用)
(5)WebUI项目引用:Application、Shared(其实这里Application引用了Shared,Web项目无需再添加引用)通常情况下,WebAPI和WebUI项目二者只选选择一个,因此新项目创建好之后,删除其中一个。
(6)Application.UnitTests项目引用:Application
(7)Domain.UnitTests项目引用:Domain
(8)Web.IntegrationTests项目引用:Web
在实际模板中,针对ServiceCollection和ApplicationBuilder写了很多扩展方法,用于一些常见组件的注册,例如Swagger、CAP(Event Bus)、Redis Client、健康检查EndPoints、KeyCloak鉴权启用等等。开发者只需要根据需要在配置文件中添加或移除对应部分的config即可,这些扩展方法会根据配置文件中是否有这部分的config来判断是否需要注册。因此,大部分情况下,小组的开发者要做的仅仅是做减法。比如,如果项目只用到了Cache没有用到EventBus,那么只需要在配置中移除EventBus的部分即可,不用改任何一句代码。又如,通常大部分项目里我们只会保留一个UnitTest类库,而不是让三个都在项目中,因为我们的精力不足以写三种类型的Test项目。但是某些大一点的项目,对质量有要求的,我们可能会写两中类型的Test项目。
模板上传Nuget仓库
这里我们主要通过将其发布为一个Nuget包上传到企业内部的Nuget仓库,然后客户端可以通过安装这个nuget包将其添加到Visual Studio中的项目模板中。
当然,也可以直接通过dotnet new命令直接通过模板创建新项目。
-- install
dotnet new install CleanArchitectureTemplate
-- uninstall
dotnet new uninstall CleanArchitectureTemplate
同理,当将开发模板发布了新的nuget包,客户端也可以通过更新nuget包的方式将模板进行更新,以便下次可以使用新的模板进行项目的开发。
-- check
dotnet new --update-check
-- apply
dotnet new --update-apply
关于如何通过Nuget上传开发模板,可以参考Microsoft的这一篇文档:https://learn.microsoft.com/zh-cn/dotnet/core/tools/custom-templates
小结
本文介绍了DDD分层架构的背景、整洁架构的概念,随后分享了一个基于我所在小组的实际情况的一个整洁架构的模板案例。在实际情况中,ABP vNext也是一不错的选择,对DDD有兴趣应用的建议仔细看看。
参考代码
GitHub:https://github.com/EdisonChou/CleanArchitectureTemplate
参考资料
Jason Taylor,《Clean Architecture with .NET Core: Gettting Started》
欧创新,极客时间《DDD实战课》
Jacky Fei,《基于ASP.NET 6的整洁架构》
Alexander Zhao,《.NET Core整洁架构入门》
Denny Zhang,《领域驱动架构及其演变历史》
聊聊 ASP.NET 6 整洁架构开发模板的更多相关文章
- 基于ASP.NET Core 6.0的整洁架构
大家好,我是张飞洪,感谢您的阅读,我会不定期和你分享学习心得,希望我的文章能成为你成长路上的垫脚石,让我们一起精进. 本节将介绍基于ASP.NET Core的整洁架构的设计理念,同时基于理论落地的代码 ...
- asp.net mvc+jquery easyui开发实战教程之网站后台管理系统开发4- 后台模板html页面创建
上一篇教程<asp.net mvc+jquery easyui开发实战教程之网站后台管理系统开发3-登录模块开发>完成了本项目的登录模块,登录后就需要进入后台管理首页了,需要准备一个后台模 ...
- 如何结合整洁架构和MVP模式提升前端开发体验 - 整体架构篇
本文不详细介绍什么是整洁架构以及 MVP 模式,自行查看文章结尾相关链接文章. 整洁架构粗略介绍 下图为整洁架构最原始的结构图: Entities/Models:实体层,官方说法就是封装了企业里最通用 ...
- 如何结合整洁架构和MVP模式提升前端开发体验(二) - 代码实现篇
上一篇文章介绍了整体架构,接下来说说怎么按照上图的分层结构实现下面的增删改查的功能. 代码结构 vue userManage └── List ├── api.ts ├── EditModal │ ├ ...
- ASP.Net的两种开发模式
一.ASP.Net的两种开发模式 1.1 ASP.Net WebForm的开发模式 (1)处理流程 在传统的WebForm模式下,我们请求一个例如http://www.aspnetmvc.com/bl ...
- 解析ASP.NET WebForm和Mvc开发的区别
因为以前主要是做WebFrom开发,对MVC开发并没有太深入的了解.自从来到创新工场的新团队后,用的技术都是自己以前没有接触过的,比如:MVC 和EF还有就是WCF,压力一直很大.在很多问题都是不清楚 ...
- 解析ASP.NET WebForm和Mvc开发的区别 分类: ASP.NET 2013-12-29 01:59 11738人阅读 评论(5) 收藏
因为以前主要是做WebFrom开发,对MVC开发并没有太深入的了解.自从来到创新工场的新团队后,用的技术都是自己以前没有接触过的,比如:MVC 和EF还有就是WCF,压力一直很大.在很多问题都是不清楚 ...
- 用VS2010创建三层架构开发模式及三层架构的研究
三层架构的研究 三层体系结构的概念 用户界面表示层(USL) 业务逻辑层(BLL) 数据访问层(DAL) BLL将USL与DAL隔开了,并且加入了业务规则 各层的作用 1:数据数据访问层:主要是对 ...
- [转]解析ASP.NET WebForm和Mvc开发的区别
因为以前主要是做WebFrom开发,对MVC开发并没有太深入的了解.自从来到创新工场的新团队后,用的技术都是自己以前没有接触过的,比如:MVC 和EF还有就是WCF,压力一直很大.在很多问题都是不清楚 ...
- .NET/ASP.NET MVC(模块化开发AraeRegistration)
阅读目录: 1.开篇介绍 2.AreaRegistration注册路由(传递路由上下文进行模块化注册) 1]开篇介绍 ASP.NET Routing 路由功能非常强大,设计的也很巧妙:如果说ASP.N ...
随机推荐
- 驱动开发:探索DRIVER_OBJECT驱动对象
本章将探索驱动程序开发的基础部分,了解驱动对象DRIVER_OBJECT结构体的定义,一般来说驱动程序DriverEntry入口处都会存在这样一个驱动对象,该对象内所包含的就是当前所加载驱动自身的一些 ...
- 使用二进制方式安装Docker
长期使用安装工具进行安装docker,今天用二进制方式手动安装一下docker环境. 二进制包下载地址:https://download.docker.com/linux/static/stable/ ...
- Java设计模式 —— 单例模式
6 单例模式 6.1 单例模式概述 Singleton Patter:确保一个类只有一个实例,并提供一个全局访问点来访问这个唯一实例. 单例模式有3个要点: 该类只能有一个实例 该类必须自行创建这个实 ...
- 【Java SE】集合
1.java集合框架 使用Array存储对象有一定的弊端.java集合就是一种容器,动态地存储多个对象,存储主要是内存层面的存储,不涉及到持久化的存储(txt,avi,数据库). ①一旦初始化好,数组 ...
- 道德与社会问题简报 #3: Hugging Face 上的道德开放性
使命: 开放和优秀的机器学习 在我们的使命中,我们致力于推动机器学习 (ML) 的民主化,我们在研究如何支持 ML 社区工作并有助于检查危害和防止可能的危害发生.开放式的发展和科学可以分散力量,让许多 ...
- 从原理聊JVM(一):染色标记和垃圾回收算法
作者:京东科技 康志兴 1 JVM运行时内存划分 1.1 运行时数据区域 • 方法区 属于共享内存区域,存储已被虚拟机加载的类信息.常量.静态变量.即时编译器编译后的代码等数据.运行时常量池,属于方法 ...
- antv x6 神奇的图片边框
昨天才把html节点中的图片转成base格式的,今天就发现一个用户体验的问题:那么是啥呢?就是我从左侧的树形菜单中拖拽节点的时候(鼠标按下也是同样问题),发现节点的图片区域那里会出现一个边框,持续时间 ...
- vCenter报错:Log Disk Exhaustion on 10
vCenter报错:Log Disk Exhaustion on 10 1.问题现象: 巡检时发现 vCenter Server 中,错误显示为:Log Disk Exhaustion on 10(字 ...
- 【学习笔记】【题解】树形依赖 DP 选做
地址:https://www.cnblogs.com/FReQuenter5156/p/shuxingyilaidp.html/ 简介 这类背包本质上是分组背包问题. 将一个节点的每一棵子树看作一组, ...
- 百度飞桨(PaddlePaddle)-数字识别
手写数字识别任务 用于对 0 ~ 9 的十类数字进行分类,即输入手写数字的图片,可识别出这个图片中的数字. 使用 pip 工具安装 matplotlib 和 numpy python -m pip i ...