【ABP】项目示例(2)——聚合根和实体
聚合根和实体
在上一章节中,已经完成了项目搭建的前置准备,在这一章节中,实现领域层的聚合根和实体
创建名称为General.Backend.Domain的标准类库,分别新建名称为Entities、Services、IRepositories和Specifications的文件夹,用于存放实体和聚合根、领域服务、仓储接口和规约。
本项目使用ABP相关的Nuget包的版本为8.3.0,为保持版本一致,后续其他ABP相关的Nuget包都使用该版本。
在程序包管理控制台选中General.Backend.Domain,执行以下命令安装ABP领域相关的Nuget包。
Install-Package Volo.Abp.Ddd.Domain -v 8.3.0
新建名称为GeneralDomainModule的领域模块类
public class GeneralDomainModule : AbpModule
{
public override void ConfigureServices(ServiceConfigurationContext context)
{
}
}
创建名称为General.Backend.Domain.Shared的标准类库,分别新建名称为Enums和Consts的文件夹,用于存放枚举和常量。
在程序包管理控制台选中General.Backend.Domain.Shared,执行以下命令安装ABP领域共享相关的Nuget包。
Install-Package Volo.Abp.Ddd.Domain.Shared -v 8.3.0
新建名称为GeneralDomainSharedModule的领域共享模块类
public class GeneralDomainSharedModule : AbpModule
{
}
General.Backend.Domain添加项目引用General.Backend.Domain.Shared
用户聚合
在Enums文件夹下新建名称为FrozenStatus的用户冻结状态枚举
/// <summary>
/// 冻结状态
/// </summary>
public enum FrozenStatus
{
/// <summary>
/// 未冻结
/// </summary>
UnFrozen = 1,
/// <summary>
/// 已冻结
/// </summary>
Frozen = 2
}
在Consts文件夹下新建名称为UserConsts的用户常量类,定义用户领域使用到的常量,用于数据库表配置和常规校验
public static class UserConsts
{
public const string UserTableName = "user";
public const string UserTableComment = "用户表";
public const string UserRoleTableName = "user_role";
public const string UserRoleTableComment = "用户角色表";
public const string AdminAccount = "admin";
public const string AdminName = "Admin";
public const int MaxLoginErrorCount = 5;
public const int MinAccountLength = 2;
public const int MaxAccountLength = 32;
public const int MinPasswordLength = 6;
public const int MaxPasswordLength = 32;
public const int MinNameLength = 1;
public const int MaxNameLength = 32;
public const int MaxContactLength = 64;
public const int MaxAddressLength = 64;
}
在Entities文件夹下新建名称为User的用户聚合根类和UserRole的用户角色实体类,业务逻辑为一个用户拥有一个或者多个角色,用户的角色需要通过用户聚合来管理
/// <summary>
/// 用户
/// </summary>
public class User : FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 账号
/// </summary>
public virtual string Account { get; private set; } = string.Empty;
/// <summary>
/// 密码
/// </summary>
public virtual string Password { get; private set; } = string.Empty;
/// <summary>
/// 用户名称
/// </summary>
public virtual string Name { get; private set; } = string.Empty;
/// <summary>
/// 联系方式
/// </summary>
public virtual string? Contact { get; set; }
/// <summary>
/// 地址
/// </summary>
public virtual string? Address { get; set; }
/// <summary>
/// 登录错误次数
/// </summary>
public virtual byte LoginErrorCount { get; private set; }
/// <summary>
/// 是否冻结(1:未冻结,2:已冻结)
/// </summary>
public virtual FrozenStatus IsFrozen { get; private set; }
/// <summary>
/// 用户关联角色列表
/// </summary>
public virtual ICollection<UserRole> UserRoles { get; private set; } = [];
protected User()
{
}
public User(
Guid id,
[NotNull] string account,
[NotNull] string password,
[NotNull] string name,
[CanBeNull] string? contact = null,
[CanBeNull] string? address = null)
{
Id = id;
SetAccount(account);
SetPassword(password);
SetName(name);
IsFrozen = FrozenStatus.UnFrozen;
Contact = Check.Length(contact, nameof(contact), UserConsts.MaxContactLength);
Address = Check.Length(address, nameof(address), UserConsts.MaxAddressLength);
}
private void SetAccount([NotNull] string account)
{
Account = Check.Length(account, nameof(account), UserConsts.MaxAccountLength, UserConsts.MinAccountLength)!;
}
internal virtual void SetPassword([NotNull] string password)
{
Password = Check.Length(password, nameof(password), UserConsts.MaxPasswordLength, UserConsts.MinPasswordLength)!;
}
public virtual void SetName([NotNull] string name)
{
Name = Check.Length(name, nameof(name), UserConsts.MaxNameLength, UserConsts.MinNameLength)!;
}
internal virtual bool CheckPassword([NotNull] string password)
{
Check.NotNullOrEmpty(password, nameof(password));
if (Password != password)
{
TryFrozen();
return false;
}
UnFrozen();
return true;
}
public virtual void UnFrozen()
{
LoginErrorCount = 0;
IsFrozen = FrozenStatus.UnFrozen;
}
private bool TryFrozen()
{
var isSuccess = false;
if (Account != UserConsts.AdminAccount)
{
LoginErrorCount += 1;
if (LoginErrorCount >= UserConsts.MaxLoginErrorCount)
{
IsFrozen = FrozenStatus.Frozen;
isSuccess = true;
}
}
return isSuccess;
}
public virtual void SetRoles(ICollection<UserRole> userRoles)
{
UserRoles = userRoles;
}
}
/// <summary>
/// 用户角色
/// </summary>
public class UserRole : Entity<Guid>, IHasCreationTime
{
/// <summary>
/// 用户Id
/// </summary>
public virtual Guid UserId { get; private set; }
/// <summary>
/// 角色Id
/// </summary>
public virtual Guid RoleId { get; private set; }
/// <summary>
/// 创建时间
/// </summary>
public virtual DateTime CreationTime { get; private set; }
protected UserRole()
{
}
internal UserRole(
Guid id,
Guid userId,
Guid roleId)
{
Id = id;
UserId = userId;
RoleId = roleId;
}
}
角色聚合
在Entities文件夹下新建名称为Role的角色聚合根类和RoleMenus的角色菜单实体类,业务逻辑为一个角色拥有一个或者多个菜单,用户拥有的菜单需要通过角色聚合来管理
/// <summary>
/// 角色
/// </summary>
public class Role : FullAuditedAggregateRoot<Guid>
{
/// <summary>
/// 编码
/// </summary>
public virtual string Code { get; private set; } = string.Empty;
/// <summary>
/// 名称
/// </summary>
public virtual string Name { get; private set; } = string.Empty;
/// <summary>
/// 描述
/// </summary>
public virtual string? Remark { get; set; }
/// <summary>
/// 角色关联菜单列表
/// </summary>
public virtual ICollection<RoleMenu> RoleMenus { get; private set; } = [];
protected Role()
{
}
public Role(
Guid id,
[NotNull] string code,
[NotNull] string name,
[CanBeNull] string? remark = null)
{
Id = id;
Code = Check.Length(code, nameof(code), RoleConsts.MaxCodeLength, RoleConsts.MinCodeLength)!;
SetName(name);
Remark = Check.Length(remark, nameof(remark), RoleConsts.MaxRemarkLength);
}
internal virtual void SetName(
[NotNull] string name)
{
Name = Check.Length(name, nameof(name), UserConsts.MaxNameLength, UserConsts.MinNameLength)!;
}
public virtual void SetMenus(ICollection<RoleMenu> roleMenus)
{
RoleMenus = roleMenus;
}
}
/// <summary>
/// 角色菜单
/// </summary>
public class RoleMenu : Entity<Guid>, IHasCreationTime
{
/// <summary>
/// 角色Id
/// </summary>
public virtual Guid RoleId { get; private set; }
/// <summary>
/// 菜单编码
/// </summary>
public virtual string MenuCode { get; private set; } = string.Empty;
/// <summary>
/// 创建时间
/// </summary>
public virtual DateTime CreationTime { get; private set; }
protected RoleMenu()
{
}
internal RoleMenu(
Guid id,
Guid roleId,
[NotNull] string menuCode)
{
Id = id;
RoleId = roleId;
MenuCode = Check.Length(menuCode, nameof(menuCode), RoleConsts.MaxMenuCodeLength, RoleConsts.MinMenuCodeLength)!;
}
}
同样地在Consts文件夹下新建名称为RoleConsts的角色常量类,定义角色领域使用到的常量,用于数据库表配置和常规校验
public class RoleConsts
{
public const string RoleTableName = "role";
public const string RoleTableComment = "角色表";
public const string RoleMenuTableName = "role_menu";
public const string RoleMenuTableComment = "角色菜单表";
public const string AdminRoleCode = "Admin";
public const string AdminRoleName = "Admin";
public const int MaxCodeLength = 32;
public const int MinCodeLength = 1;
public const int MaxNameLength = 32;
public const int MinNameLength = 1;
public const int MaxMenuCodeLength = 64;
public const int MinMenuCodeLength = 1;
public const int MaxRemarkLength = 255;
}
菜单聚合
在Entities文件夹下新建名称为Menu的菜单聚合根类
/// <summary>
/// 菜单
/// </summary>
public class Menu : BasicAggregateRoot<Guid>, IHasCreationTime
{
/// <summary>
/// 编码
/// </summary>
public virtual string Code { get; private set; } = string.Empty;
/// <summary>
/// 父编码
/// </summary>
public virtual string ParentCode { get; private set; } = string.Empty;
/// <summary>
/// 名称
/// </summary>
public virtual string Name { get; private set; } = string.Empty;
/// <summary>
/// 类型
/// </summary>
public virtual string Type { get; private set; } = string.Empty;
/// <summary>
/// 层级
/// </summary>
public virtual int Level { get; private set; }
/// <summary>
/// 图标
/// </summary>
public virtual string? Icon { get; private set; }
/// <summary>
/// 路由地址
/// </summary>
public virtual string? UrlAddress { get; private set; }
/// <summary>
/// 组件地址
/// </summary>
public virtual string? ComponentAddress { get; private set; }
/// <summary>
/// 排序
/// </summary>
public virtual int Sort { get; private set; }
/// <summary>
/// 创建时间
/// </summary>
public virtual DateTime CreationTime { get; private set; }
/// <summary>
/// 子菜单
/// </summary>
public List<Menu> SubMenu { get; private set; } = [];
protected Menu()
{
}
public Menu(
[NotNull] string code,
[NotNull] string parentCode,
[NotNull] string name,
[NotNull] string type,
int level,
int sort,
[CanBeNull] string? icon = null,
[CanBeNull] string? urlAddress = null,
[CanBeNull] string? componentAddress = null)
{
Code = Check.Length(code, nameof(code), MenuConsts.MaxCodeLength, MenuConsts.MinCodeLength)!;
ParentCode = Check.Length(parentCode, nameof(parentCode), MenuConsts.MaxParentCodeLength, MenuConsts.MinParentCodeLength)!;
Name = Check.Length(name, nameof(name), MenuConsts.MaxNameLength, MenuConsts.MinNameLength)!;
Type = Check.Length(type, nameof(type), MenuConsts.MaxTypeLength, MenuConsts.MinTypeLength)!;
Level = level;
Sort = sort;
Icon = Check.Length(icon, nameof(icon), MenuConsts.MaxIconLength);
UrlAddress = Check.Length(urlAddress, nameof(urlAddress), MenuConsts.MaxUrlAddressLength);
ComponentAddress = Check.Length(componentAddress, nameof(componentAddress), MenuConsts.MaxComponentAddressLength);
}
private Menu(
[NotNull] string code)
{
Code = Check.Length(code, nameof(code), MenuConsts.MaxCodeLength, MenuConsts.MinCodeLength)!;
}
internal static Menu BuildRoot()
{
return new Menu(
MenuConsts.MenuRootCode);
}
internal static Menu BuildCatalog(
[NotNull] string code,
[NotNull] string name,
[NotNull] string icon,
[NotNull] string urlAddress,
int sort)
{
return new Menu(
code,
MenuConsts.MenuRootCode,
name,
MenuConsts.CatalogTypeName,
1,
sort,
icon,
urlAddress);
}
internal static Menu BuildMenu(
[NotNull] string code,
[NotNull] string parentCode,
[NotNull] string name,
[NotNull] string icon,
[NotNull] string urlAddress,
[NotNull] string componentAddress,
int sort)
{
return new Menu(
code,
parentCode,
name,
MenuConsts.MenuTypeName,
2,
sort,
icon,
urlAddress,
componentAddress);
}
internal static Menu BuildFunc(
[NotNull] string code,
[NotNull] string parentCode,
[NotNull] string name,
int sort)
{
return new Menu(
code,
parentCode,
name,
MenuConsts.FuncTypeName,
3,
sort);
}
public void SetSubMenu(
[NotNull] List<Menu> subMenus)
{
Check.NotNull(subMenus, nameof(subMenus));
foreach (var parentCode in subMenus.Select(menu => menu.ParentCode))
{
Check.Equals(Code, parentCode);
}
SubMenu = subMenus;
}
}
同样地在Consts文件夹下新建名称为MenuConsts的菜单常量类
public class MenuConsts
{
public const string MenuTableName = "menu";
public const string MenuTableComment = "菜单表";
public const string MenuRootCode = "root";
public const string CatalogTypeName = "C";
public const string MenuTypeName = "M";
public const string FuncTypeName = "F";
public const int MaxCodeLength = 64;
public const int MinCodeLength = 1;
public const int MaxParentCodeLength = 64;
public const int MinParentCodeLength = 1;
public const int MaxNameLength = 32;
public const int MinNameLength = 1;
public const int MaxTypeLength = 8;
public const int MinTypeLength = 1;
public const int MaxIconLength = 32;
public const int MaxUrlAddressLength = 64;
public const int MaxComponentAddressLength = 64;
}
上面已经对用户、角色和菜单领域创建了相应的聚合根,相关的业务逻辑以聚合根和实体中方法来实现
解决方案的目录结构现如下
在下一章节中,使用仓储作为领域模型和数据模型的桥梁,将领域模型持久化到数据库中的数据模型中
【ABP】项目示例(2)——聚合根和实体的更多相关文章
- 基于ABP实现DDD--聚合和聚合根实践
在下面的例子中涉及Repository.Issue.Label.User这4个聚合根,接下来以Issue聚合为例进行分析,其中Issue聚合是由Issue[聚合根].Comment[实体].Iss ...
- 基于ABP落地领域驱动设计-02.聚合和聚合根的最佳实践和原则
目录 前言 聚合 聚合和聚合根原则 包含业务原则 单个单元原则 事务边界原则 可序列化原则 聚合和聚合根最佳实践 只通过ID引用其他聚合 用于 EF Core 和 关系型数据库 保持聚合根足够小 聚合 ...
- 初识ABP vNext(11):聚合根、仓储、领域服务、应用服务、Blob存储
Tips:本篇已加入系列文章阅读目录,可点击查看更多相关文章. 目录 前言 开始 聚合根 仓储 领域服务 BLOB存储 应用服务 单元测试 模块引用 最后 前言 在前两节中介绍了ABP模块开发的基本步 ...
- DDD:使用EntityFramework的话,如果只为聚合根设计仓储,其它实体如何处理?
背景 DDD中只有聚合根可以有仓储,仓储负责整个聚合持久化的相关生命周期,在不使用工作单元或POCO的情况下,我们可以让Order内部直接调用DAL操作OrderItem.我们也可以让Order跟踪所 ...
- DDD的实体、值对象、聚合根的基类和接口:设计与实现
1 前置阅读 在阅读本文章之前,你可以先阅读: 什么是DDD 2 实现值对象 值对象有两个主要特征:它们没有任何标识.它们是不可变的. 我们举个例子:小明是"浙江宁波"人,小红也是 ...
- 从壹开始微服务 [ DDD ] 之六 ║聚合 与 聚合根 (下)
前言 哈喽大家周二好,上次咱们说到了实体与值对象的简单知识,相信大家也是稍微有些了解,其实实体咱们平时用的很多了,基本可以和数据库表进行联系,只不过值对象可能不是很熟悉,值对象简单来说就是在DDD领域 ...
- DDD中聚合、聚合根的含义以及作用
聚合与聚合根的含义 聚合: 聚合往往是一些实体为了某项业务而聚类在一起形成的集合 , 举个例子, 社会是由一个个的个体组成的,象征着我们每一个人.随着社会的发展,慢慢出现了社团.机构.部门等组织,我们 ...
- DDD之4聚合和聚合根
聚合就是归类的意思,把同类事物统一处理: 聚合根也就是最抽象,最普遍的特性: 背景 领域建模的过程回顾: 那么问题来了? 为什么要在限界上下文和实体之间增加聚合和聚合根的概念,即作用是什么? 如何设计 ...
- 关于ABP聚合根类AggregateRoot的思考
AggregateRoot和Entity的区别 AggregateRoot继承于Entity,并实现了IGeneratesDomainEvents接口 public class AggregateRo ...
- Abp 领域事件简单实践 <四> 聚合根的领域事件
聚合根有个 DomainEvents 属性. 首先聚合根是一个实体.这个实体的仓储有变化(增删改)的时候,会触发这个DomainEvents 里的事件.就像EventBus.Trigger一样. pu ...
随机推荐
- Three.js入门-常见几何体
这章节,我们将介绍 Three.js 中常见的几何体(Geometry),包括立方体.球体.圆柱体.平面.圆环.圆锥体等.几何体是构建 3D 模型的基础元素,通过不同的几何体可以创建出各种形状的物体. ...
- YashanDB演讲实录|别彬彬:金融科技对智能化创新系统的机遇与路径
本文为"2024国产数据库创新生态大会"深算院采石矶.钓鱼城系统技术总监别彬彬的演讲实录分享,主题为 <金融科技对智能化创新系统的机遇与路径>,欢迎阅读. 各位领导.嘉 ...
- 解决Your project does not reference问题
错误现象 vs编译时,报错: Your project does not reference ".NETFramework,Version=v##" framework. Add ...
- 安卓导出已安装app的apk
安卓导出已安装应用APK 有时候想看看别人的APK里面的资源文件或者是逆向,首先就得先搞到APK文件 两种方法获取手机上已安装应用的APK文件 通过adb命令 首先把目标手机连接上电脑 在终端输入 a ...
- Docker 使用 buildx 构建多种系统架构支持的镜像
在 Docker 19.03+ 版本中可以使用 docker buildx build 命令使用 BuildKit 构建镜像.该命令支持 --platform 参数可以同时构建支持多种系统架构的 Do ...
- Netty 缓存buffer介绍及使用
每当你需要传输数据时,它必须包含一个缓冲区.Java NIO API 自带的缓冲区类是相当有限的,没有经过优化,使用 JDK 的ByteBuffer 操作更复杂.缓冲区是一个重要的组建,它是 API的 ...
- Qt音视频开发07-合并音视频文件
一.前言 之前已经把音视频分开存储了对应的文件,因为这个需求特别少,当然确实有部分用户是需要把音视频分开存储,但是毕竟是很少数,绝大部分的用户都是音视频合并到一个MP4文件,所以如果要合并到一个文件, ...
- UML之类与类图
在所有项目中,类都是最常见的UML模型元素(当然,不可否认,很多项目还没画出类图就直接进入编码实现的阶段了).类是UML模型与具体实现代码之间的桥梁,随着对UML建模的深入了解,我们也会发现,类(确切 ...
- 在命令中输入信息创建maven项目
参考链接: 1.使用命令行创建maven web项目 2.Maven 三种archetype说明 3.maven创建项目时在generating project in interactive mode ...
- 记录实现倒计时的方法,配合按钮的disabled
记录一个自己实现倒计时的方法,现在可以网上有很多插件,自己实现记录一下 // 倒计时 countDown() { this.disabled = true let number = 60 this.c ...