百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 编程网 > 正文

聊聊 ASP.NET 6 整洁架构开发模板

yuyutoo 2024-10-26 16:10 1 浏览 0 评论

大家好,我是Edison。

之前看了一些整洁架构(CleanArchitecture)的文章,也简单写了一个基于整洁架构的ASP.NET 6开发模板在玩。这里就仅仅抛个砖,案例主要以自己根据小组实际情况做了一些裁剪,不具有通用的应用性,大家看看就好。

整洁架构的产生背景
微服务架构让DDD(领域驱动)焕发了第二春,在DDD的推动下,DDD的分层架构被逐渐推上了舞台。
DDD的分层架构就有好多种,例如整洁架构、CQRS和六边形架构等等,每种架构模式虽然提出的时代和背景不同,但其核心理念都是为了设计出“高内聚低耦合”的架构,从而能够实现架构的演进。

DDD分层架构

在欧创新老师的《DDD实战课》中,给出了一个优化后的DDD四层架构,我们可以从下面这张图中看到,从上到下分别是:用户接口层、应用层、领域层和基础层。
与传统的三层架构不同,DDD四层架构的重点在于引入了一个领域层
领域层的作用是实现企业核心业务逻辑,通过各种校验手段保证业务的正确性。领域层主要体现领域模型的业务能力,它用来表达业务概念、业务状态和业务规则。
领域层包含:聚合根、实体、值对象、领域服务等领域模型中的领域对象。
对于领域层,领域模型的业务逻辑主要由实体和领域服务来实现。对于实体,一般建议采用充血模型来实现所有与之相关的业务功能。对于领域服务,一般当单个实体不能实现某些功能时,领域服务才会出马,组合聚合内的多个实体来实现复杂的业务逻辑。
下图中展示了传统的三层架构与DDD四层架构的对应关系:

整洁架构简单介绍
简而言之,整洁架构是组织软件体系结构的原则,可以轻松面对未来的不确定性,方便代码的重构。同时,它可以帮助我们为特定的领域模型构建服务,从而为将来可能的微服务体系结构做好准备。
在Jason Taylor的这篇文章中《Clean Architecture with .NET Core: Gettting Started》(https://jasontaylor.dev/clean-architecture-getting-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中的解决方案目录如下图所示:

最终的依赖关系如下:
(1)Domain类库只引用Shared类库(即CrossCutting)。
(2)Application类库引用:Domain、Infrastructure、Shared
这里我们没有让Application不依赖于Infrastructure,是因为我们的DB技术栈已经固定,而且也比较习惯于在Infrastructure里边定义Repository、Gateway的接口(interface和实现(implement),而不是在Application里面定义这些interface,Infrastructure通过引用Application来做实现。
(3)Infrastructure类库引用:Domain、Shared
(4)Web项目引用:ApplicationShared(其实这里Application引用了Shared,Web项目无需再添加引用)
(5)WebUI项目引用:ApplicationShared(其实这里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命令直接通过模板创建新项目。
-- installdotnet new install CleanArchitectureTemplate-- uninstalldotnet new uninstall CleanArchitectureTemplate
同理,当将开发模板发布了新的nuget包,客户端也可以通过更新nuget包的方式将模板进行更新,以便下次可以使用新的模板进行项目的开发。
-- checkdotnet new --update-check-- applydotnet new --update-apply
关于如何通过Nuget上传开发模板,可以参考Microsoft的这一篇文档:
https://learn.microsoft.com/zh-cn/dotnet/core/tools/custom-templates

小结

本文介绍了DDD分层架构的背景、整洁架构的概念,随后分享了一个基于我所在小组的实际情况的一个整洁架构的模板案例。在实际情况中,在基于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,《领域驱动架构及其演变历史》

相关推荐

jQuery VS AngularJS 你更钟爱哪个?

在这一次的Web开发教程中,我会尽力解答有关于jQuery和AngularJS的两个非常常见的问题,即jQuery和AngularJS之间的区别是什么?也就是说jQueryVSAngularJS?...

Jquery实时校验,指定长度的「负小数」,小数位未满末尾补0

在可以输入【负小数】的输入框获取到焦点时,移除千位分隔符,在输入数据时,实时校验输入内容是否正确,失去焦点后,添加千位分隔符格式化数字。同时小数位未满时末尾补0。HTML代码...

如何在pbootCMS前台调用自定义表单?pbootCMS自定义调用代码示例

要在pbootCMS前台调用自定义表单,您需要在后台创建表单并为其添加字段,然后在前台模板文件中添加相关代码,如提交按钮和表单验证代码。您还可以自定义表单数据的存储位置、添加文件上传字段、日期选择器、...

编程技巧:Jquery实时验证,指定长度的「负小数」

为了保障【负小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【负小数】的方法。HTML代码<inputtype="text"class="forc...

一篇文章带你用jquery mobile设计颜色拾取器

【一、项目背景】现实生活中,我们经常会遇到配色的问题,这个时候去百度一下RGB表。而RGB表只提供相对于的颜色的RGB值而没有可以验证的模块。我们可以通过jquerymobile去设计颜色的拾取器...

编程技巧:Jquery实时验证,指定长度的「正小数」

为了保障【正小数】的正确性,做成了通过Jquery,在用户端,实时验证指定长度的【正小数】的方法。HTML做成方法<inputtype="text"class="fo...

jquery.validate检查数组全部验证

问题:html中有多个name[],每个参数都要进行验证是否为空,这个时候直接用required:true话,不能全部验证,只要这个数组中有一个有值就可以通过的。解决方法使用addmethod...

Vue进阶(幺叁肆):npm查看包版本信息

第一种方式npmviewjqueryversions这种方式可以查看npm服务器上所有的...

layui中使用lay-verify进行条件校验

一、layui的校验很简单,主要有以下步骤:1.在form表单内加上class="layui-form"2.在提交按钮上加上lay-submit3.在想要校验的标签,加上lay-...

jQuery是什么?如何使用? jquery是什么功能组件

jQuery于2006年1月由JohnResig在BarCampNYC首次发布。它目前由TimmyWilson领导,并由一组开发人员维护。jQuery是一个JavaScript库,它简化了客户...

django框架的表单form的理解和用法-9

表单呈现...

jquery对上传文件的检测判断 jquery实现文件上传

总体思路:在前端使用jquery对上传文件做部分初步的判断,验证通过的文件利用ajaxFileUpload上传到服务器端,并将文件的存储路径保存到数据库。<asp:FileUploadI...

Nodejs之MEAN栈开发(四)-- form验证及图片上传

这一节增加推荐图书的提交和删除功能,来学习node的form提交以及node的图片上传功能。开始之前需要源码同学可以先在git上fork:https://github.com/stoneniqiu/R...

大数据开发基础之JAVA jquery 大数据java实战

上一篇我们讲解了JAVAscript的基础知识、特点及基本语法以及组成及基本用途,本期就给大家带来了JAVAweb的第二个知识点jquery,大数据开发基础之JAVAjquery,这是本篇文章的主要...

推荐四个开源的jQuery可视化表单设计器

jquery开源在线表单拖拉设计器formBuilder(推荐)jQueryformBuilder是一个开源的WEB在线html表单设计器,开发人员可以通过拖拉实现一个可视化的表单。支持表单常用控件...

取消回复欢迎 发表评论: