(精华)2020年8月22日 ABP vNext DTO在应用层的使用

我们继续应用层的开发,首先创建负责在应用层和展示层之间传递数据的对象,也就是DTO。

使用DTO的原因

为什么需要DTO呢?有如下几个原因。

  1. 隔离领域层与表示层,使领域层和表示层可以独立演化,互相不受影响。
  2. 数据隐藏,领域层的某些数据需要对表示层隐藏(比如用户密码),在定义DTO时,可以不设置隐藏字段的映射,实现数据隐藏。DTO只返回表示层需要的数据,不多也不少。
  3. 避免序列化问题。领域对象中会带有循环引用,比如诗人Poet会引用诗Poems,而诗Poem中又引用了诗人Poet,这种循环引用在序列化时会出现问题。

ABP vNext中DTO和实体的对应关系

ABP vNext使用映射Profile定义DTO和实体之间的映射关系,这与以前版本使用AutoMap标签是不同的,映射文件如下:
以前使用如下:

[AutoMapFrom(typeof(Products))]
    public class ProductDto:EntityDto
    {
     
        public string ProductName {
      get; set; }
        public string ProductImage {
      get; set; }
        public double Price {
      get; set; }
        public double OrgPrice {
      get; set; }
        public string Decoration {
      get; set; }
        public string Sizes {
      get; set; }
        public int ClickTimes {
      get; set; }
        public int SaleTimes {
      get; set; }
        public string DetailText {
      get; set; }
    }

现在使用如下:

using AutoMapper;
using ZL.AbpNext.Poem.Core.Poems;

namespace ZL.AbpNext.Poem.Application.Poems
{
     
    public class PoemAppAutoMapperProfile : Profile
    {
     
        public PoemAppAutoMapperProfile()
        {
     
            CreateMap<Poet, PoetDto>();
            CreateMap<Core.Poems.Poem, PoemDto>();
            CreateMap<Category, CategoryDto>();
            CreateMap<CategoryPoem, CategoryPoemDto>();
        }
    }
}

我们已经创建了领域层,并且使用EF实现了对数据库的访问,我们还创建了用于数据交换的DTO,现在继续应用层的开发。我们需要实现如下基本需求:

  • 诗人查询:按姓名进行模糊查询
  • 根据id获取诗人数据。
  • 诗查询:
    按诗人进行查询;按关键字在标题进行模糊查询;按分类进行查询,如果分类是多个,就查询属于所有分类的诗。比如,如果查询条件是“唐诗三百首”和“五言诗”,那么结果应该是唐诗三百首中的五言诗。
  • 根据id获取诗数据。
  • 分类列表:列出所有分类。
  • 分类的增、删:可以增加和删除分类。
  • 查询某一首诗的所有分类。

上面的功能能够基本完成诗和诗人查询分类的功能。

上述功能的定义在接口IPoemAppService中定义,在PoemAppService中实现。IPoemAppService代码如下:


using System.Collections.Generic;
using Volo.Abp.Application.Dtos;
using Volo.Abp.Application.Services;

namespace ZL.AbpNext.Poem.Application.Poems
{
     
    public interface IPoemAppService:IApplicationService
    {
     
        /// 
        /// 获取诗人分页
        /// 
        /// 
        /// 
        PagedResultDto<PoetDto> GetPagedPoets(PagedResultRequestDto dto);

        /// 
        /// 查询诗人,按名字模糊查询
        /// 
        /// 
        /// 
        PagedResultDto<PoetDto> SearchPoets(SearchPoetDto dto);

        /// 
        /// 获取诗的分页查询
        /// 
        /// 
        /// 
        PagedResultDto<PoemDto> GetPagedPoems(PagedResultRequestDto dto);


        /// 
        /// 按条件查询诗,条件是关键字(模糊查询),作者(精确查询),分类(属于所有分类)
        /// 
        /// 
        /// 
        PagedResultDto<PoemDto> SearchPoems(SearchPoemDto dto);

        /// 
        /// 增加分类,如果已经存在,不增加,返回-1,如果增加成功,返回新增记录的id
        /// 
        /// 
        /// 
        CategoryDto AddCategory(CategoryDto category);

        /// 
        /// 删除分类
        /// 
        /// 
        void DeleteCategory(CategoryDto category);

        /// 
        /// 分类列表
        /// 
        /// 
        List<CategoryDto> GetAllCategories();

        /// 
        /// 将诗关联到分类
        /// 
        /// 
        void AddPoemToCategory(CategoryPoemDto categoryPoem);

        /// 
        /// 解除诗和分类的关联
        /// 
        /// 
        void RemovePoemFromCategory(CategoryPoemDto categoryPoem);


        List<CategoryPoemDto> GetCategoryPoems();

        /// 
        /// 列出诗的分类
        /// 
        /// 
        /// 
        List<CategoryDto> GetPoemCategories(int poemid);

        /// 
        /// 列出分类的诗
        /// 
        /// 
        /// 
        List<PoemDto> GetPoemsOfCategory(int categoryid);


        PoetDto AddPoet(PoetDto poet);
    }
}

现在我们可以编写PoemAppService,这个类的定义如下:

public class PoemAppService : ApplicationService, IPoemAppService
    {
     
        private readonly IRepository<Core.Poems.Poem> _poemRepository;
        private readonly IRepository<Category> _categoryRepository;
        private readonly IRepository<Poet> _poetRepository;
        private readonly IRepository<CategoryPoem> _categoryPoemRepository;
        public PoemAppService(IRepository<Core.Poems.Poem> poemRepository
            , IRepository<Category> categoryRepository
            , IRepository<Poet> poetRepository
            , IRepository<CategoryPoem> categoryPoemRepository)
        {
     
            _poemRepository = poemRepository;
            _categoryRepository = categoryRepository;
            _poetRepository = poetRepository;
            _categoryPoemRepository = categoryPoemRepository;
        }

这里我们使用Abp提供的通用的IRepository,编写一般的方法可以满足需求,比如增加,删除等等:

        public CategoryDto AddCategory(CategoryDto category)
        {
     
            var cate = _categoryRepository.FirstOrDefault(o => o.CategoryName == category.CategoryName);

            if (cate == null)
            {
     
                cate= _categoryRepository.InsertAsync(new Category {
      CategoryName = category.CategoryName },true).Result;
            }
            return ObjectMapper.Map<Category,CategoryDto>(cate); 
            
        }

        public List<CategoryDto> GetAllCategories()
        {
     
            return ObjectMapper.Map<List<Category>, List<CategoryDto>>(_categoryRepository.ToList());
        }

        public void DeleteCategory(CategoryDto category)
        {
     
                var cat = _categoryRepository.FirstOrDefault(o => o.Id == category.Id);
                if (cat != null)
                {
     
                    _categoryRepository.DeleteAsync(cat, true);
                }
         }

这里需要说明的是,Abp 将ApplicationService中的方法作为工作单元处理,所以不需要像在一开始控制台应用中那样显示使用UnitWork。

当我们进行稍微复杂一些的开发时,发现通用的IRepository不够用了,比如,如果获取是的分类,需要写成这样:

public List<CategoryDto> GetPoemCategories(int poemid)
{
     
    var lst = _categoryPoemRepository.Where(p => p.PoemId == poemid);
    var categories = new List<Category>();
    foreach (var cp in lst)
    {
     
        var cate = _categoryRepository.GetAsync(o => o.Id == cp.CategoryId).Result;
        categories.Add(cate);
    }

    return ObjectMapper.Map<List<Category>, List<CategoryDto>>(categories);
}

对于更复杂的一些功能,比如模糊查询,使用缺省的IRepository甚至无法实现。下节我们开发定制的Repository。

你可能感兴趣的:(#,ABP,vNext微服务框架)