![](/img/trans.png)
[英]How to map Expression<Func<TEntity, bool>> to Expression<Func<TDbEntity, bool>>
[英]Convert Complex Expression<Func<TDTO>,bool> to Expression<Func<TEntity>,bool>
我的框架中有 DTOmodel。 BLL 和 View 根本看不到 EntiyModel。 我想在層之間傳輸和轉換復雜表達式。
這是 BLL 中的一種方法,就像這樣......
//// BLL(Service)
Public PersonDTO getAll(Expression<Func<PersonDTO, bool>> whereCondition)
{
return _repository.getAll(whereCondition);
}
// DLL(Repository)
Public PersonDTO getAll(Expression<Func<PersonDTO, bool>> whereCondition)
{
Expression<Func<Person, bool>> NewCondition = ?/ How Convert DTOwhereCondition ???
return DataContext.Persons(NewCondition);
}
///// 我想像這樣在 PersonDTO 上創建復雜的表達式:
var persons = serive.getPersons(i => i.PersonDetailsDTO.Count == 3);
///// 我的課程
public class Person
{
public Int32 Id { get; set; }
public String FirstName { get; set; }
public String LastName { get; set; }
public List<PersonDetail> PersonDetails { get; set; }
}
public class PersonDTO
{
public Int32 Id { get; set; }
public String FirstName { get; set; }
public String LastName { get; set; }
public List<PersonDetailDTO> PersonDetailsDTO { get; set; }
}
這是最基本的解決方案。 如果您需要更多功能,也許可以從那里繼續進行。 到目前為止,它允許有選擇地提供委托來解析映射的屬性名稱,但是屬性類型必須相同。
用法:
var mappedExpression =
ExpressionMapper<Person>.MapFrom(originalDtoExpression, null);
// or:
var mappedExpression =
ExpressionMapper<Person>.MapFrom(originalDtoExpression,
entityPropertyName => someMapper.ResolveDtoPropertyName(entityPropertyName));
實現方式:
public class ExpressionMapper<TTarget> : ExpressionVisitor
{
protected ExpressionMapper(Type sourceType, Func<string, string> resolveTargetMemberNameFunc)
{
this.SourceType = sourceType;
this.ResolveTargetMemberNameFunc = resolveTargetMemberNameFunc ?? (cur => cur);
}
public Func<string, string> ResolveTargetMemberNameFunc { get; private set; }
public Type SourceType { get; private set; }
public static Expression MapFrom<TSource>(Expression sourceExpression, Func<string, string> resolveTargetMemberNameFunc)
{
return new ExpressionMapper<TTarget>(typeof(TSource), resolveTargetMemberNameFunc).Visit(sourceExpression);
}
public static Expression<Func<TTarget, bool>> MapFrom<TSource>(Expression<Func<TSource, bool>> sourcePredicateExpression, Func<string, string> resolveTargetMemberNameFunc)
{
return Expression.Lambda<Func<TTarget, bool>>(
MapFrom<TSource>(sourcePredicateExpression.Body, resolveTargetMemberNameFunc),
Expression.Parameter(typeof(TTarget), sourcePredicateExpression.Parameters[0].Name));
}
protected override Expression VisitMember(MemberExpression node)
{
var sourceMemberName = node.Member.Name;
var targetNode = this.Visit(node.Expression);
var targetMemberName = this.ResolveTargetMemberNameFunc(sourceMemberName);
var targetMember = targetNode.Type.GetMember(targetMemberName).FirstOrDefault();
if (targetMember == null)
{
throw new NotSupportedException(String.Format("The source type '{0}' cannot be mapped to the target type '{1}', because that target type has no member '{2}'.", node.Expression.Type.Name, targetNode.Type.Name, targetMemberName));
}
return Expression.MakeMemberAccess(targetNode, targetMember);
}
protected override Expression VisitParameter(ParameterExpression node)
{
if (node.Type == this.SourceType)
{
return Expression.Parameter(typeof(TTarget), node.Name);
}
return node;
}
}
我使用 AutoMapper 將使用 DTO 的表達式轉換為實體。 表達式映射是 AutoMapper.Extensions.ExpressionMapping 包的一部分。 當您調用 AddAutoMapper(它本身是 IServiceCollection 的擴展方法)時,您本質上必須調用 AddExpressionMapping
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.