我的泛型类中有以下方法:
// This is the class declaration
public abstract class BaseService<TEntity, TKey> : IBaseService<TEntity, TKey> where TEntity : class, IEntity<TKey>
// The Method
public IQueryable<TEntity> GetActive()
{
if (typeof(IActivable).IsAssignableFrom(typeof(TEntity)))
{
return this.repository.Get().Cast<IActivable>()
.Where(q => q.Active)
.Cast<TEntity>();
}
else
{
return this.Get();
}
}
这是界面:
public interface IActivable
{
bool Active { get; set; }
}
基本上,TEntity
是一个实体 (POCO) 类,如果它们具有,则可以实现 IActivableActive
财产。我想要该方法返回所有具有的记录Active
值为真。但是,我有这个错误:
无法将类型“WebTest.Models.Entities.Product”转换为类型
'数据.IActivable'。 LINQ to Entities 仅支持铸造 EDM
原始类型或枚举类型。
我明白为什么会出现这个错误。但 SO 上的文章对我的情况没有任何有效的解决方案。是否可以实现Cast
,或者任何其他方式?注意:我不想转换为IEnumerable
,我想保留IQueryable
.
EF 表达式解析器无需强制转换即可工作,但是如果没有强制转换,您将无法编译 C# 代码(C# 会抱怨它不知道TEntity
has an Active
财产)。解决方案是:针对 c# 编译器进行强制转换,而不针对 EF 表达式解析器进行强制转换。
因此,如果您确定(您正在检查if
,所以你是)该对象实现IActivable
,您可以使用强制转换创建表达式(用于编译),然后在运行时删除 EF 的强制转换(这是不必要的)。对于您的具体情况:
public IQueryable<TEntity> GetActive()
{
if (typeof(IActivable).IsAssignableFrom(typeof(TEntity)))
{
Expression<Func<TEntity, bool>> getActive = x => ((IActivable)x).Active;
getActive = (Expression<Func<TEntity, bool>>)RemoveCastsVisitor.Visit(getActive);
return this.repository.Get().Where(getActive);
}
else
{
return this.Get();
}
}
表达式访问者的实现如下:
internal class RemoveCastsVisitor : ExpressionVisitor
{
private static readonly ExpressionVisitor Default = new RemoveCastsVisitor();
private RemoveCastsVisitor()
{
}
public new static Expression Visit(Expression node)
{
return Default.Visit(node);
}
protected override Expression VisitUnary(UnaryExpression node)
{
if (node.NodeType == ExpressionType.Convert
&& node.Type.IsAssignableFrom(node.Operand.Type))
{
return base.Visit(node.Operand);
}
return base.VisitUnary(node);
}
}
它只是检查是否需要进行转换:如果实际值已经实现了它要转换到的类型,它只会从表达式中删除转换,并且 EF 将正确选择它。
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)