## 应用服务最佳实践 & 约定 * **推荐** 为每个 **聚合根** 创建一个应用服务. ### 应用服务接口 * **推荐** 在 **application.contracts**层中为每一个应用服务定义一个`接口`. * **推荐** 继承 `IApplicationService` 接口 . * **推荐** 接口名称使用`AppService` 后缀 (如: `IProductAppService`). * **推荐** 为服务创建输入输出DTO(数据传输对象). * **不推荐** 服务中含有返回实体的方法. * **推荐** 根据[DTO 最佳实践](Data-Transfer-Objects.md)定义DTO. #### 输出 * **避免** 为相同或相关实体定义过多的输出DTO. 为实体定义 **基础** 和 **详细** DTO. ##### 基础DTO **推荐** 为实体定义一个**基础**DTO. - 直接包含实体中所有的**原始属性**. - 例外: 出于**安全**原因,可以**排除**某些属性(像 `User.Password`). - 包含实体中所有**子集合**, 每个集合项都是一个简单的**关系DTO**. 示例: ```c# public class IssueDto : FullAuditedEntityDto { public string Title { get; set; } public string Text { get; set; } public Guid? MilestoneId { get; set; } public Collection Labels { get; set; } } public class IssueLabelDto { public Guid IssueId { get; set; } public Guid LabelId { get; set; } } ``` ##### 详细DTO **Do** 如果实体持有对其他聚合根的引用,那么应该为其定义**详细**DTO. * 直接包含实体中所有的 **原始属性**. - 例外-1: 出于**安全**原因,可以**排除**某些属性(像 `User.Password`). - 例外-2: **推荐** 排除引用属性(如上例中的 `MilestoneId`). 为其添加引用属性的详细信息. * 为每个引用属性添加其**基本DTO** . * 包含实体的**所有子集合**, 集合中的每项都是相关实体的基本DTO. 示例: ````C# public class IssueWithDetailsDto : FullAuditedEntityDto { public string Title { get; set; } public string Text { get; set; } public MilestoneDto Milestone { get; set; } public Collection Labels { get; set; } } public class MilestoneDto : EntityDto { public string Name { get; set; } public bool IsClosed { get; set; } } public class LabelDto : EntityDto { public string Name { get; set; } public string Color { get; set; } } ```` #### 输入 * **不推荐** 在输入DTO中定义未在服务类中使用的属性. * **不推荐** 在应用服务方法之间共享输入DTO. * **不推荐** 继承另一个输入DTO类. * **可以** 继承自抽象基础DTO类, 并以这种方式在不同的DTO之间共享一些属性. 但是在这种情况下需要非常小心, 因为更新基础DTO会影响所有相关的DTO和服务方法. 所以避免这样做是一种好习惯. #### 方法 * **推荐** 为异步方法使用 **Async** 后缀. * **不推荐** 在方法名中重复实体的名称. * 例如: 在 `IProductAppService` 中定义`GetAsync(...)` 而不是 `GetProductAsync(...)` . ##### 获取单一实体 * **推荐** 使用 `GetAsync` 做为**方法名**. * **推荐** 使用id做为方法参数. * 返回 **详细DTO**. 示例: ````C# Task GetAsync(Guid id); ```` ##### 获取实体集合 * **推荐** 使用 `GetListAsync` 做为**方法名**. * **推荐** 如果需要获取单个DTO可以使用参数进行 **过滤**, **排序** 和 **分页**. * **推荐** 尽可能让过滤参数可选. * **推荐** 将排序与分页属性设置为可选, 并且提供默认值. * **推荐** 限制最大页数大小 (基于性能考虑). * **推荐** 返回 **详细DTO**集合. 示例: ````C# Task> GetListAsync(QuestionListQueryDto queryDto); ```` ##### 创建一个新实体 * **推荐** 使用 `CreateAsync` 做为**方法名**. * **推荐** 使用**专门的输入DTO**来创建实体. * **推荐** 使用 **data annotations** 进行输入验证. * 尽可能在**领域**之间共享常量(通过**domain shared** package定义的常量). * **推荐** 只需要创建实体的**最少**信息, 但是提供了其他可选属性. 示例**方法**: ````C# Task CreateAsync(CreateQuestionDto questionDto); ```` 输入**DTO**: ````C# public class CreateQuestionDto { [Required] [StringLength(QuestionConsts.MaxTitleLength, MinimumLength = QuestionConsts.MinTitleLength)] public string Title { get; set; } [StringLength(QuestionConsts.MaxTextLength)] public string Text { get; set; } //Optional public Guid? CategoryId { get; set; } //Optional } ```` ##### 更新已存在的实体 - **推荐** 使用 `UpdateAsync` 做为**方法名**. - **推荐** 使用**专门的输入DTO**来更新实体. - **推荐** 获取实体的id做为分离的原始参数. 不要包含更新DTO. - **推荐** 使用 **data annotations** 进行输入验证. - 尽可能在**领域**之间共享常量(通过**domain shared** package定义的常量). - **推荐** 返回更新实体的**详细**DTO. 示例: ````C# Task UpdateAsync(Guid id, UpdateQuestionDto updateQuestionDto); ```` ##### 删除已存在的实体 - **推荐** 使用 `DeleteAsync` 做为**方法名**. - **推荐** 使用原始参数 id. 示例: ````C# Task DeleteAsync(Guid id); ```` ##### 其他方法 * **可以** 定义其他方法以对实体执行操作. 示例: ````C# Task VoteAsync(Guid id, VoteType type); ```` 此方法为试题投票并返回试题的当前分数. ### 应用服务实现 * **推荐** 开发**完全独立于web层**的应用层. * **推荐** 在**应用层**实现应用服务接口. * **推荐** 使用命名约定. 如: 为 `IProductAppService` 接口创建 `ProductAppService` 类. * **推荐** 继承自 `ApplicationService` 基类. #### 使用仓储 * **推荐** 使用专门设计的仓储 (如 `IProductRepository`). * **不推荐** 使用泛型仓储 (如 `IRepository`). #### 查询数据 * **不推荐** 在应用程序服务方法中使用linq/sql查询来自数据库的数据. 让仓储负责从数据源执行linq/sql查询. #### 操作/删除 实体 * **推荐** 总是从数据库中获取所有的相关实体以对他们执行操作. #### 使用其他应用服务 * **不推荐** 使用相同 **模块/应用程序** 的其他应用服务. 相反; * 使用领域层执行所需的任务. * 提取新类并在应用程序服务之间共享, 在必要时代码重用. * **可以** 在以下情况下使用其他应用服务; * 它们是另一个模块/微服务的一部分. * 当前模块仅引用已使用模块的application contracts.