我认为有一个简单的方法可以解决它,但它是一个hack。让我再次强调一下——它确实是一个hack。你可以试试这个:
Repository.Get<TEntity, TKey>(e => (object)e.Id == (object)EntityId);
上面的代码通常不应该工作。在 CLR 世界中,这些值将被装箱并通过引用进行比较。即使装箱的值相同,引用也会不同,因此结果将是错误的。但是 EF 查询不由 CLR 执行,而是转换为 SQL。结果,查询将被翻译为:WHERE Id = {EntityId}
这是您需要的。同样,使用它需要了解这些东西的工作原理和原因,并且可能有点冒险。但是,由于存在 hack,因此应该有更清洁的解决方案。事实上,干净的(这里不是简单的解决方案)是手动构建上述表达式。这是一个示例(对不起,我没有完全使用您的实体):
private static TEntity GetEntity<TEntity, TKey>(Expression<Func<TEntity, TKey>> property, TKey keyValue)
where TKey : struct
where TEntity : BaseEntity<TKey>
{
using (var ctx = new Context2())
{
var query = Filter(ctx.Set<TEntity>(), property, keyValue);
return query.First();
}
}
private static IQueryable<TEntity> Filter<TEntity, TProperty>(IQueryable<TEntity> dbSet,
Expression<Func<TEntity, TProperty>> property,
TProperty value)
where TProperty : struct
{
var memberExpression = property.Body as MemberExpression;
if (memberExpression == null || !(memberExpression.Member is PropertyInfo))
{
throw new ArgumentException("Property expected", "property");
}
Expression left = property.Body;
Expression right = Expression.Constant(value, typeof (TProperty));
Expression searchExpression = Expression.Equal(left, right);
var lambda = Expression.Lambda<Func<TEntity, bool>>(Expression.Equal(left, right),
new ParameterExpression[] {property.Parameters.Single()});
return dbSet.Where(lambda);
}
请注意,在 Filter 方法中,我构建了一个可以编写的过滤器表达式。在这个例子中,有效的查询看起来像这样 DbSet().Where(e => e.Id == idValue).First() (看起来类似于上面的 hack)但是你可以在这个查询之上使用其他 linq 运算符(包括在Filter方法的结果上调用Filter方法来按多个条件过滤)
我将实体和上下文定义如下:
public class BaseEntity<TKey> where TKey : struct
{
public TKey Id { get; set; }
}
public class EntityWithIntKey : BaseEntity<int>
{
public string Name { get; set; }
}
public class EntityWithGuidKey : BaseEntity<Guid>
{
public string Name { get; set; }
}
public class Context2 : DbContext
{
public DbSet<EntityWithIntKey> EntitiesWithIntKey { get; set; }
public DbSet<EntityWithGuidKey> EntitiesWithGuidKey { get; set; }
}
您可以像这样调用 GetEntity 方法:var e2 = GetEntity(e => e.Id, guidKey);