我们继续应用层的开发,首先创建负责在应用层和展示层之间传递数据的对象,也就是DTO。
为什么需要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,现在继续应用层的开发。我们需要实现如下基本需求:
上面的功能能够基本完成诗和诗人查询分类的功能。
上述功能的定义在接口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。