这篇文章主要介绍“.NET 6开发TodoList应用怎么实现数据塑形”,在日常操作中,相信很多人在.NET 6开发TodoList应用怎么实现数据塑形问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”.NET 6开发TodoList应用怎么实现数据塑形”的疑惑有所帮助!接下来,请跟着小编一起来学习吧!
需求
在查询的场景中,还有一类需求不是很常见,就是在前端请求中指定返回的字段,所以关于搜索的最后一个主题我们就来演示一下关于数据塑形(Data Shaping)。
目标
实现数据塑形搜索请求。
原理与思路
对于数据塑形来说,我们需要定义一些接口和泛型类实现来完成通用的功能,然后修改对应的查询请求,实现具体的功能。
实现
定义通用接口和泛型类实现
IDataShaper.cs
using System.Dynamic;namespace TodoList.Application.Common.Interfaces;public interface IDataShaper<T>{ IEnumerable<ExpandoObject> ShapeData(IEnumerable<T> entities, string fieldString); ExpandoObject ShapeData(T entity, string fieldString);}
并实现通用的功能:
DataShaper.cs
using System.Dynamic;using System.Reflection;using TodoList.Application.Common.Interfaces;namespace TodoList.Application.Common;public class DataShaper<T> : IDataShaper<T> where T : class{ public PropertyInfo[] Properties { get; set; } public DataShaper() { Properties = typeof(T).GetProperties(BindingFlags.Public | BindingFlags.Instance); } public IEnumerable<ExpandoObject> ShapeData(IEnumerable<T> entities, string? fieldString) { var requiredProperties = GetRequiredProperties(fieldString); return GetData(entities, requiredProperties); } public ExpandoObject ShapeData(T entity, string? fieldString) { var requiredProperties = GetRequiredProperties(fieldString); return GetDataForEntity(entity, requiredProperties); } private IEnumerable<PropertyInfo> GetRequiredProperties(string? fieldString) { var requiredProperties = new List<PropertyInfo>(); if (!string.IsNullOrEmpty(fieldString)) { var fields = fieldString.Split(',', StringSplitOptions.RemoveEmptyEntries); foreach (var field in fields) { var property = Properties.FirstOrDefault(pi => pi.Name.Equals(field.Trim(), StringComparison.InvariantCultureIgnoreCase)); if (property == null) { continue; } requiredProperties.Add(property); } } else { requiredProperties = Properties.ToList(); } return requiredProperties; } private IEnumerable<ExpandoObject> GetData(IEnumerable<T> entities, IEnumerable<PropertyInfo> requiredProperties) { return entities.Select(entity => GetDataForEntity(entity, requiredProperties)).ToList(); } private ExpandoObject GetDataForEntity(T entity, IEnumerable<PropertyInfo> requiredProperties) { var shapedObject = new ExpandoObject(); foreach (var property in requiredProperties) { var objectPropertyValue = property.GetValue(entity); shapedObject.TryAdd(property.Name, objectPropertyValue); } return shapedObject; }}
定义扩展方法
为了使我们的Handle方法调用链能够直接应用,我们在Application/Extensions中新增一个DataShaperExtensions:
DataShaperExtensions.cs
using System.Dynamic;using TodoList.Application.Common.Interfaces;namespace TodoList.Application.Common.Extensions;public static class DataShaperExtensions{ public static IEnumerable<ExpandoObject> ShapeData<T>(this IEnumerable<T> entities, IDataShaper<T> shaper, string? fieldString) { return shaper.ShapeData(entities, fieldString); }}
然后再对我们之前写的MappingExtensions静态类中添加一个方法:
MappingExtensions.cs
// 省略其他...public static PaginatedList<TDestination> PaginatedListFromEnumerable<TDestination>(this IEnumerable<TDestination> entities, int pageNumber, int pageSize){ return PaginatedList<TDestination>.Create(entities, pageNumber, pageSize); }
添加依赖注入
在Application的DependencyInjection.cs中添加依赖注入:
DependencyInjection.cs
// 省略其他services.AddScoped(typeof(IDataShaper<>), typeof(DataShaper<>));
修改查询请求和Controller接口
我们在上一篇文章实现排序的基础上增加一个字段用于指明数据塑形字段并对应修改Handle方法:
GetTodoItemsWithConditionQuery.cs
using System.Dynamic;using AutoMapper;using AutoMapper.QueryableExtensions;using MediatR;using TodoList.Application.Common.Extensions;using TodoList.Application.Common.Interfaces;using TodoList.Application.Common.Mappings;using TodoList.Application.Common.Models;using TodoList.Application.TodoItems.Specs;using TodoList.Domain.Entities;using TodoList.Domain.Enums;namespace TodoList.Application.TodoItems.Queries.GetTodoItems;public class GetTodoItemsWithConditionQuery : IRequest<PaginatedList<ExpandoObject>>{ public Guid ListId { get; set; } public bool? Done { get; set; } public string? Title { get; set; } // 前端指明需要返回的字段 public string? Fields { get; set; } public PriorityLevel? PriorityLevel { get; set; } public string? SortOrder { get; set; } = "title_asc"; public int PageNumber { get; set; } = 1; public int PageSize { get; set; } = 10;}public class GetTodoItemsWithConditionQueryHandler : IRequestHandler<GetTodoItemsWithConditionQuery, PaginatedList<ExpandoObject>>{ private readonly IRepository<TodoItem> _repository; private readonly IMapper _mapper; private readonly IDataShaper<TodoItemDto> _shaper; public GetTodoItemsWithConditionQueryHandler(IRepository<TodoItem> repository, IMapper mapper, IDataShaper<TodoItemDto> shaper) { _repository = repository; _mapper = mapper; _shaper = shaper; } public Task<PaginatedList<ExpandoObject>> Handle(GetTodoItemsWithConditionQuery request, CancellationToken cancellationToken) { var spec = new TodoItemSpec(request); return Task.FromResult( _repository .GetAsQueryable(spec) .ProjectTo<TodoItemDto>(_mapper.ConfigurationProvider) .AsEnumerable() // 进行数据塑形和分页返回 .ShapeData(_shaper, request.Fields) .PaginatedListFromEnumerable(request.PageNumber, request.PageSize) ); }}
对应修改Controller:
TodoItemController.cs
[HttpGet]public async Task<ApiResponse<PaginatedList<ExpandoObject>>> GetTodoItemsWithCondition([FromQuery] GetTodoItemsWithConditionQuery query){ return ApiResponse<PaginatedList<ExpandoObject>>.Success(await _mediator.Send(query));}
验证
启动Api项目,执行查询TodoItem的请求:
请求
响应
我们再把之前讲到的过滤和搜索添加到请求里来:
请求
响应
到此,关于“.NET 6开发TodoList应用怎么实现数据塑形”的学习就结束了,希望能够解决大家的疑惑。理论与实践的搭配能更好的帮助大家学习,快去试试吧!若想继续学习更多相关知识,请继续关注编程网网站,小编会继续努力为大家带来更多实用的文章!