1

抱歉标题,我找不到合适的描述。

我有四层:

  • 核心层:包含 DTO、服务和存储库的接口。
  • 业务层:包含处理业务逻辑的“服务”。
  • 数据访问层:包含处理数据库访问和实体到 DTO 的转换的存储库。
  • 表示层:UI 的东西

我遇到了一个我不知道如何最好地解决的问题。我正在向数据库异步添加一个实体,如下所示:

// The AdministrationRate has an ID property, Entity Framework does treat the property as an Identity and does increment it.
var adminRate = new AdministrationRate() {FosterChildID = fosterChild.ID};
await adminRateService.AddAdministrationRate(adminRate);

管理费率服务:

public async Task AddAdministrationRate(AdministrationRate administrationRate) => await repo.AddAdministrationRate(administrationRate);

管理率存储库:

 //I use a generic repository to avoid code repition.
 //Notice the DTO to entity conversion. ID is not set in the conversion.
 public async Task AddAdministrationRate(AdministrationRate administrationRate) => await Add(administrationRate.ToEntity());

存储库:

public async Task Add(TEntity entity)
{
    using (var db = new DatabaseContext())
    {
        db.Entry(entity).State = EntityState.Added;
        await db.SaveChangesAsync();
    }
}

这确实有效,问题是在添加之后,新生成的实体 id 没有反映在 DTO (DTO.ID = 0) 中。

我不可能返回更新后的实体,因为它需要我将实体转换为我的表示层中的 DTO,以使其始终保持异步。

由于通用存储库,我也不能只返回 ID。

我真的不想摆脱通用存储库,因为它非常有用,但是我真的看不到另一种在不更改数据库的情况下安全地执行此操作的方法(我不能这样做)。

那么,我该怎么办?

4

2 回答 2

0

答: 我找到了两个解决这个问题的方法,后者是我实际使用的解决方案。


解决方案1:

我发现我可以让 Add 方法返回实体的 ID,方法是将通用存储库的 Add 方法标记为虚拟,然后在特定存储库之一中覆盖它,最后从调用代码中捕获结果。这当然意味着更多的代码,但我也不会在错误的层中从实体转换为 DTO(这不是一个选项)。


解决方案2:

显然,await关键字异步返回任务的结果类型,而不是像那样阻塞它result,所以我能够从我的存储库转换并返回一个 DTO:

管理率存储库:

public async Task<AdministrationRate> AddAdministrationRate(AdministrationRate administrationRate) => (await Add(administrationRate.ToEntity())).ToModel();

我想如果有人可以评论我是否需要使我的ToEntity()ToModel()方法异步,以及如何做到这一点。

编辑:哦,没有注意到有人发帖。我会看看你的建议 Akos,谢谢。

于 2017-08-30T12:59:37.657 回答
0

这是我会尝试的几个想法。不确定它们中的任何一个是否适合您的需求,但可能会让您开始(或至少开始讨论:)):

1个;为您的 DTO 引入一个接口,例如IDto. 然后,您可以将通用存储库更改为返回Task<IDto>而不是 Task。将实体转换为Add方法内的具体 Dto,通过IDto引用返回,然后在服务中将其转换回。

2;如果您有像IRepository<TEntity>, 或之类的通用存储库Repository<TEntity>,那么您可以尝试将另一个参数引入通用存储库并将其设为IRepository<TEntity,TDto>. 这样,您可以保持架构的通用性,但也可以Task<Dto>Add方法返回。

3;你可以走得更远:创建一个IDto<TEntity>界面。然后你可以有一个User实体和一个UserDto:IDto<User>. 然后您可以将Add方法的返回类型更改为Task<IDto<TEntity>>. 或者,再次使用双参数存储库IRepository<TEntity,TDto>,并添加约束,这样就TDto必须实现IDto<TEntity>.

于 2017-08-30T12:46:57.030 回答