本文主要介绍了多应用层的问题,包括原因和实现。通过理解介绍了如何区分领域逻辑和应用逻辑,哪些是正确的实践,哪些是不推荐的或者错误的实践。

一.多应用层的问题

1.多应用层介绍

  不知道你们是否会遇到一种情况,通过ABP构建了一个后端的API项目,刚开始是为Web端项目(比如,Vue)提供后端接口服务的,随着项目的发展和业务的复杂,增加了移动端的App,或者公众号、小程序等,这样不仅要为Web端提供API服务,而且还需要为移动端的App,或者公众号、小程序等提供API服务。这个场景就是多应用层的问题。也就是说你现在需要构建一个有多个应用程序的系统了:

  • Web应用程序:比如使用的ASP.NET Core MVC技术,主要用来给用户展示产品,游客模式是可以查看产品的,并不需要登录和验证。
  • 后端管理应用程序:比如前端是Vue,后端的ASP.NET Core API,主要用来对产品进行增删改等操作。
  • 移动应用程序:比如前端是uni-app,后端是ASP.NET Core API,通过REST接口和后端通信。

2.多应用层例子

  由于业务的复杂性,每个应用系统都有自己的不同应用服务方法,不同的输入和输出DTO,不用的认证和授权规则等,所以如果把所有的业务逻辑都融入到一个应用系统中,就会让系统更难开发、维护和测试,并导致潜在的Bug风险。因此,为每个应用程序创建单独的应用层,并且它们都是用单个领域层来共享核心领域逻辑。为了更加具体的说明,为每种应用程序类型创建不同的.csproj项目:

  • 后端管理应用程序:IssueTracker.Admin.Application和IssueTracker.Admin.Application.Contracts项目
  • Web应用程序:IssueTracker.Public.Application和IssueTracker.Public.Application.Contracts项目
  • 移动应用程序:IssueTracker.Mobile.Application和IssueTracker.Mobile.Application.Contracts项目

二.如何区分领域逻辑和应用逻辑

通常,DDD中的业务逻辑包括领域逻辑和应用逻辑。领域逻辑由系统的核心领域规则组成,而应用程序逻辑实现特定于应用程序的用例。话虽这样说,但是并不容易区分什么是领域逻辑和应用逻辑。

1.在领域服务层中创建组织Organization

下面通过在领域服务中创建Organization这个例子,来尽可能简要说明:

  1. public class OrganizationManager:DomainService
  2. {
  3. private readonly IRepository<Organization> _organizationRepository; //Organization的仓储
  4. private readonly ICurrentUser _currentUser; //当前用户
  5. private readonly IAuthorizationService _authorizationService; //Authorization的服务
  6. private readonly IEmailSender _emailSender; //邮件发送服务
  7. // 公共构造函数,依赖注入
  8. public OrganizationManager(IRepository<Organization> organizationRepository, ICurrentUser currentUser, IAuthorizationService authorizationService, IEmailSender emailSender)
  9. {
  10. _organizationRepository=organizationRepository;
  11. _currentUser=currentUser;
  12. _authorizationService=authorizationService;
  13. _emailSender=emailSender;
  14. }
  15. // 创建一个新的组织
  16. public async Task<Organization> CreateAsync(string name)
  17. {
  18. // 如果组织存在同名,那么抛出异常[正确]
  19. if(await _organizationRepository.AnyAsync(x=>x.Name==name))
  20. {
  21. throw new BusinessException("IssueTracking:DuplicateOrganizationName");
  22. }
  23. // 检查是否拥有创建的权限[错误]
  24. await _authorizationService.CheckAsync("OrganizationCreationPermission");
  25. // 记录⽇志[错误]
  26. Logger.LogDebug($"Creating organization {name} by {_currentUser.UserName}");
  27. // 创建一个新的组织
  28. var organization = new Organization();
  29. // 发送邮件进行提醒[错误]
  30. await _emailSender.SendAsync("systemadmin@issuetracking.com", "新组织", "新组织名称:"+name);
  31. // 返回一个组织实例
  32. return organization;
  33. }
  34. }
  • 领域服务不做权限验证,权限验证放在应用层来做
  • 用户概念是应用层或展示层的相关概念,记录日志不应该包含当前用户的用户名
  • 创建一个新的组织,并发送邮件,这个业务逻辑也应该放在应用层

2.在应用层中使用领域服务创建组织Organization

  1. public class OrganizationAppService:ApplicationService
  2. {
  3. private readonly OrganizationManager _organizationManager; //组织的领域服务
  4. private readonly IPaymentService _paymentService; //支付服务
  5. private readonly IEmailSender _emailSender; //邮件服务
  6. // 公共构造函数,依赖注入
  7. public OrganizaitonAppService(OrganizationManager organizationManager, IPaymentService paymentService, IEmailSender emailSender)
  8. {
  9. _organizationManager=organizationManager;
  10. _paymentService=paymentService;
  11. _emailSender=emailSender;
  12. }
  13. // 创建组织
  14. [UnitOfWork][正确] //工作单元,用于提交事务
  15. [Authorize("OrganizationCreationPermission")][正确]
  16. public async Task<Organization> CreateAsync(CreateOrganizationDto input)
  17. {
  18. // ⽀付组织的费⽤[正确]
  19. await _paymentService.ChargeAsync(CurrentUser.Id, GetOrganizationPrice());
  20. // 通过领域服务,创建一个新的组织实例
  21. var organization = await _organizationManager.CreateAsync(input.Name);
  22. // 保存和更新组织到数据库中[正确]
  23. await _organizationManager.InsertAsync(organization);
  24. // 发送提醒邮件[正确]
  25. await _emailSender.SendAsync("systemadmin@issuetracking.com", "新组织", "新组织名称:"+name);
  26. //返回实例[错误]
  27. return organization;
  28. }
  29. private double GetOrganizationPrice()
  30. {
  31. return 42;//Gets form somewhere...
  32. }
  33. }

应用服务层的输入和输出参数都是DTO,不能返回实体。至于为什么不将支付放在领域服务中,只能说业务重要也不一定放在领域服务中,详细原因说明参考[1]。

参考文献:

[1]基于ABP Framework实现领域驱动设计:https://url39.ctfile.com/f/2501739-616007877-f3e258?p=2096 (访问密码: 2096)

基于ABP实现DDD--领域逻辑和应用逻辑的更多相关文章

  1. 基于ABP落地领域驱动设计-06.正确区分领域逻辑和应用逻辑

    目录 系列文章 领域逻辑和应用逻辑 多应用层 示例:正确区分应用逻辑和领域逻辑 学习帮助 系列文章 基于ABP落地领域驱动设计-00.目录和前言 基于ABP落地领域驱动设计-01.全景图 基于ABP落 ...

  2. 基于事件驱动的DDD领域驱动设计框架分享(附源代码)

    原文:基于事件驱动的DDD领域驱动设计框架分享(附源代码) 补充:现在再回过头来看这篇文章,感觉当初自己偏激了,呵呵.不过没有以前的我,怎么会有现在的我和现在的enode框架呢?发现自己进步了真好! ...

  3. 基于ABP落地领域驱动设计-00.目录和小结

    <实现领域驱动设计> -- 基于 ABP Framework 实现领域驱动设计实用指南 翻译缘由 自 ABP vNext 1.0 开始学习和使用该框架,被其优雅的设计和实现吸引,适逢 AB ...

  4. 基于ABP落地领域驱动设计-04.领域服务和应用服务的最佳实践和原则

    目录 系列文章 领域服务 应用服务 学习帮助 系列文章 基于ABP落地领域驱动设计-00.目录和前言 基于ABP落地领域驱动设计-01.全景图 基于ABP落地领域驱动设计-02.聚合和聚合根的最佳实践 ...

  5. 基于ABP落地领域驱动设计-05.实体创建和更新最佳实践

    目录 系列文章 数据传输对象 输入DTO最佳实践 不要在输入DTO中定义不使用的属性 不要重用输入DTO 输入DTO中验证逻辑 输出DTO最佳实践 对象映射 学习帮助 系列文章 基于ABP落地领域驱动 ...

  6. 基于ABP落地领域驱动设计-02.聚合和聚合根的最佳实践和原则

    目录 前言 聚合 聚合和聚合根原则 包含业务原则 单个单元原则 事务边界原则 可序列化原则 聚合和聚合根最佳实践 只通过ID引用其他聚合 用于 EF Core 和 关系型数据库 保持聚合根足够小 聚合 ...

  7. 基于ABP实现DDD--领域服务、应用服务和DTO实践

      什么是领域服务呢?领域服务就是领域对象本身的服务,通常是通过多个聚合以实现单个聚合无法处理的逻辑. 一.领域服务实践 接下来将聚合根Issue中的AssignToAsync()方法[将问题分配给用 ...

  8. 基于ABP实现DDD--聚合和聚合根实践

      在下面的例子中涉及Repository.Issue.Label.User这4个聚合根,接下来以Issue聚合为例进行分析,其中Issue聚合是由Issue[聚合根].Comment[实体].Iss ...

  9. 基于ABP实现DDD--DDD相关概念

      什么是DDD呢?领域驱动设计[DDD]是一种针对复杂需求的软件开发方法.将软件实现与不断发展的模型联系起来,专注于核心领域逻辑,而不是基础设施细节.DDD适用于复杂领域和大规模应用,而不是简单的C ...

随机推荐

  1. PHP代码审计之SQL注入

    代码审计之SQL注入 SQL注入攻击(SQLInjection),是攻击者在表单中提交精心构造的sql语句,改变原来的sql语句,如果web程序没有对提交的数据经过检查,那么就会造成sql注入攻击. ...

  2. 一个Python中优雅的数据分块方法

    背景 看到这个标题你可能想一个分块能有什么难度?还值得细说吗,最近确实遇到一个有意思的分块函数,写法比较巧妙优雅,所以写一个分享. 日前在做需求过程中有一个对大量数据分块处理的场景,具体来说就是几十万 ...

  3. js运算符、 流程控制 、函数、内置对象、BOM与DOM操作

    运算符 # 1.算术运算符 var x=10; var res1=x++; '先赋值后自增1' var res2=++x; '先自增1后赋值' # 2.比较运算符 弱等于:自动转换类型 '5' == ...

  4. 关于c#多线程中的几个信号量

    信号量在c#多线程通信中主要用来向阻塞的线程传达信号从而使得阻塞线程继续执行 多线程信号(线程交互):通常是指线程必须等待一个线程或者多个线程通知交互(释放信号)才可以继续执行 在c#中信号量主要有这 ...

  5. opencv c++安装踩坑记录 file cannot create directory: /usr/local/include/opencv2. Maybe need administrative privileges

    前言 最近深度学习Ultra-Fast-Lane-Detection/INSTALL.md at master · cfzd/Ultra-Fast-Lane-Detection (github.com ...

  6. Date类的常见用法——JavaSE基础

    Date类的常见用法 Date类属于java.util包 因此需要导入Date类 Date() 分配一个Date对象,并初始化此对象为系统当前的日期和时间,可以精确到毫秒). Date(long da ...

  7. Java - 四种引用类型及应用场景

    1. 强引用 new 一个对象的时候,就是强引用 Object object = new Object(); 只要强引用存在,垃圾回收就不会回收该对象,内存不足时会抛出OOM. 2. 软引用 定义:非 ...

  8. npm版本兼容导致的npm ERR! ERESOLVE unable to resolve dependency tree

    当团队项目中,团队成员的npm包管理工具版本不一致时执行npm install报错: npm -v查看版本信息:7.x与6.x之间的兼容问题 解决方案: 一:升级或降级npm版本,保持一致npm in ...

  9. 深度学习与CV教程(13) | 目标检测 (SSD,YOLO系列)

    作者:韩信子@ShowMeAI 教程地址:http://www.showmeai.tech/tutorials/37 本文地址:http://www.showmeai.tech/article-det ...

  10. .NET中按预定顺序执行任务

    更新记录 本文迁移自Panda666原博客,原发布时间:2021年7月1日. 一.说明 在.NET中线程可以定义按先后顺序进行执行,适合部分有先后次序的业务逻辑.Task也可以按照预定义的先后顺序执行 ...