16

几天来我一直试图弄清楚这一点,但似乎很少有关于 ASP.NET MVC 的特定主题的信息。我已经在谷歌上搜索了几天,并没有真正弄清楚这个特定问题。

我有一个 3 层项目。业务、DAL 和 UI/Web 层。在 DAL 中是 dbcontext、存储库和工作单元。业务层是一个领域层,包含所有接口和 EF 模型。在业务层中,还有一个带有用于 EF 模型的 DTO 的服务层和一个访问存储库的通用存储库服务。这张图片应该有助于解释它。

我的问题是我似乎无法弄清楚如何使用 DTO 从业务层传输数据。

我为 DTO 创建了服务类。我有一个 ImageDTO 和模型,对于图像锚点也是如此。我为每个 DTO 创建了一个服务类。所以我有一个图像服务和锚点服务。这些服务继承了存储库服务,目前实现了它们自己的服务。但这就是我所得到的。由于这些服务具有通过 IoC 接收 IUnitOfWork 接口的构造函数,因此我几乎陷入了困境。

如果我直接从 UI 引用服务,一切都会正常工作,但我只是想不通如何使用 DTO 将数据从服务层传输到 UI 层,反之亦然。

我的服务层:

业务/服务/DTO

public class AnchorDto
{
      public int Id { get; set; }
      public int x1 { get; set; }
      public int y1 { get; set; }
      public int x2 { get; set; }
      public int y2 { get; set; }
      public string description { get; set; }
      public int  imageId { get; set; }
      public int targetImageId { get; set; }

      public AnchorDto(int Id, int x1, int y1, int x2, int y2, string description, int imageId, int targetImageId)
      {
          // Just mapping input to the DTO 
      }
}

public class ImageDto
{
    public int Id { get; set; }
    public string name { get; set; }
    public string title { get; set; }
    public string description { get; set; }
    public virtual IList<AnchorDto> anchors { get; set; }

    public ImageDto(int Id, string name, string title, string description, IList<AnchorDto> anchors )
    {
        // Just mapping input to DTO
    }
}

业务/服务/服务

public class RepoService<TEntity> : IRepoService<TEntity> where TEntity : class
{
    private IRepository<TEntity> repo;

    public RepoService(IUnitOfWork repo)
    {
        this.repo = repo.GetRepository<TEntity>();
    }

    public IEnumerable<TEntity> Get(
        Expression<Func<TEntity, bool>> filter = null,
        Func<IQueryable<TEntity>, IOrderedQueryable<TEntity>> orderBy = null,
        string includeProperties = "")
        {
            return repo.Get(filter, orderBy, includeProperties);
        }

        public TEntity GetByID(object id)
        {
            return repo.GetByID(id);
        }

        public void Insert(TEntity entity)
        {
            repo.Insert(entity);
        }

        public void Delete(object id)
        {
            repo.Delete(id);
        }

        public void Delete(TEntity entityToDelete)
        {
            repo.Delete(entityToDelete);
        }

        public void Update(TEntity entityToUpdate)
        {
            repo.Update(entityToUpdate);
        }
    }

Image Service,IImageService 接口目前是空的,直到我弄清楚我需要实现什么。

public class ImageService : RepoService<ImageModel>, IImageService
{
    public ImageService(IUnitOfWork repo)
        : base(repo)
    {

    }
}

目前我的控制器并没有真正工作并且没有使用服务层,所以我决定不包括任何这些。一旦我解决了这个问题,我计划使用自动映射器将 DTO 映射到 ViewModel。

所以现在,请任何知识渊博的人给我我所缺少的想法,以便我能解决这个问题?

4

1 回答 1

46

您的服务应该接收 DTO,将它们映射到业务实体并将它们发送到存储库。它还应该从存储库中检索业务实体,将它们映射到 DTO 并将 DTO 作为响应返回。因此,您的业务实体永远不会脱离业务层,只有 DTO 会这样做。

那么您的 UI\Weblayer 应该不知道业务实体。Web 层应该只知道 DTO。执行此规则非常重要,您的 UI 层不使用服务实现类(应该是私有的),只使用接口。并且服务接口不应该依赖于业务实体,而应该依赖于 DTO。

因此,您需要基于 DTO 的服务接口,并且您的基础服务类需要另一个用于 DTO 的通用参数。我喜欢有一个实体和 DTO 的基类,因此它们可以声明为:

//Your UI\presentation layer will work with the interfaces (The inheriting ones) 
//so it is very important that there is no dependency
//on the business entities in the interface, just on the DTOs!
protected interface IRepoService<TDto> 
    where TDto: DTOBase
{
    //I'm just adding a couple of methods  but you get the idea
    TDto GetByID(object id);
    void Update(TDto entityToUpdateDto)
}

//This is the interface that will be used by your UI layer
public IImageService: IRepoService<ImageDTO>
{
}

//This class and the ones inheriting should never be used by your 
//presentation\UI layer because they depend on the business entities!
//(And it is a best practice to depend on interfaces, anyway)
protected abstract class RepoService<TEntity, TDto> : IRepoService<TDto> 
    where TEntity : EntityBase
    where TDto: DTOBase
{
    ... 
}

//This class should never be used by your service layer. 
//Your UI layer should always use IImageService
//You could have a different namespace like Service.Implementation and make sure
//it is not included by your UI layer
public class ImageService : RepoService<ImageModel, ImageDto>, IImageService
{
    ...
}

然后,您需要一种在不实际实现映射的情况下将实体和 DTO 之间的映射添加到该基础服务的方法(因为它取决于每个具体实体和 DTO 类)。您可以声明执行映射的抽象方法,并且需要在每个特定服务(如ImageService)上实现。基本 RepoService 的实现如下所示:

public TDto GetByID(object id)
{
    //I'm writing it this way so its clear what the method is doing
    var entity = repo.GetByID(id);
    var dto = this.EntityToDto(entity);
    return dto;
}

public void Update(TDto entityToUpdateDto)
{
    var entity = this.DtoToEntity(entityToUpdateDto)
    repo.Update(entity);
}

//These methods will need to be implemented by every service like ImageService
protected abstract TEntity DtoToEntity(TDto dto);
protected abstract TDto EntityToDto(TEntity entity);

或者,您可以声明映射服务,添加一个具有应由您的 IOC 提供的适当映射服务的依赖项(如果您需要在不同服务上使用相同的映射,这更有意义)。RepoService 的实现如下所示:

private IRepository<TEntity> _repo;
private IDtoMappingService<TEntity, TDto> _mappingService;

public RepoService(IUnitOfWork repo, IDtoMappingService<TEntity, TDto> mapping)
{
    _repo = repo.GetRepository<TEntity>();
    _mappingService = mapping;
}

public TDto GetByID(object id)
{
    //I'm writing it this way so its clear what the method is doing
    var entity = repo.GetByID(id);
    var dto = _mappingService.EntityToDto(entity);
    return dto;
}

public void Update(TDto entityToUpdateDto)
{
    var entity = _mappingService.DtoToEntity(entityToUpdateDto)
    repo.Update(entity);
}

//You will need to create implementations of this interface for each 
//TEntity-TDto combination
//Then include them in your dependency injection configuration
public interface IDtoMappingService<TEntity, TDto>
    where TEntity : EntityBase
    where TDto: DTOBase
{
    public TEntity DtoToEntity(TDto dto);
    public TDto EntityToDto(TEntity entity);
}

在这两种情况下(抽象方法或映射服务),您都可以手动实现实体和 DTO 之间的映射,也可以使用Automapper之类的工具。但是在使用 AutoMapper 和实体框架时应该非常小心,尽管那是另一个话题!(谷歌一点点,并收集有关该主题的一些信息。作为第一个建议,请注意在加载数据时对数据库执行的查询,这样您就不会加载超出需要或发送很多查询。保存数据时请注意到您的收藏和关系)

可能是长帖子,但我希望它有帮助!

于 2013-06-01T11:19:47.697 回答