我有这个通用存储库。
/// <summary>
/// Implémentation de base d'un dépositoire pour Entity Framework.
/// </summary>
/// <remarks>Entity Framework 4.1</remarks>
/// <typeparam name="TEntite">Entité spécifique.</typeparam>
public abstract class RepositoryBase<TEntity, TKey> : IRepository<TEntity>, IDisposable
where TEntity : EntityBase<TKey>
where TKey : class
{
private readonly IContext _context;
private ObjectContext _objectContext;
private IObjectSet<TEntity> _objectSet;
protected RepositoryBase(IContext context)
{
_context = context;
_objectContext = _context.GetObjectContext();
_objectSet = _objectContext.CreateObjectSet<TEntity>();
}
/// <see cref="IRepository.cs"/>
public IEnumerable<TEntity> GetAll(Expression<Func<TEntity, object>> sortExpression)
{
if (sortExpression == null)
sortExpression = x => x.Id;
return _objectSet.OrderBy(sortExpression).AsEnumerable();
}
/// <see cref="IRepository.cs"/>
public IEnumerable<TEntity> GetAll(int maximumRows, int startRowIndex, Expression<Func<TEntity, object>> sortExpression)
{
if (sortExpression == null)
sortExpression = x => x.Id;
return _objectSet.OrderBy(sortExpression).Skip(startRowIndex).Take(maximumRows).AsEnumerable();
}
/// <see cref="IRepository.cs"/>
public TEntity SelectByKey(TKey key)
{
if (key == null)
throw new ArgumentNullException("La clé était NULL! Une clé est nécessaire pour récupérer un entité.");
return _objectSet.SingleOrDefault(x => x.Id == key);
}
这里具体实施......
public class ProductRepository : RepositoryBase<Product, int>, IProductRepository { public ProductRepository(IContext context) : base(context) { }
/// <see cref="IProductRepository.cs"/>
public Product GetByName(string name)
{
return base.First(x => x.Name == name);
}
/// <see cref="IProductRepository.cs"/>
public IEnumerable<Product> FindProduct(Specification<Product> specification)
{
throw new System.NotImplementedException();
}
}
当我创建一个特定的存储库时......它说Tkey必须是引用类型,但为什么呢?有没有办法让这个通用存储库工作?使用TKey是为了使方法SelectByKey接受密钥类型。
编辑#1:
如果我删除约束,那么我还有另一个问题......在我的SelectByKey方法中使用== as lambda表达式无法将TKey与TKey进行比较。
编辑#2:
尝试使用Equals,语法似乎没问题。
编辑#3:
在运行时等于崩溃。说Tkey(似乎是一个System.Object)不能使用看起来不合逻辑的等号,因为对象具有相同的方法。我目前无法访问真实代码,但我使用下面的代码进行了一些测试..
class Program
{
static void Main(string[] args)
{
Test<TestEntity, int> t = new Test<TestEntity, int>();
t.TestMethod(5);
}
}
class Test<TEntity, TKey>
where TEntity : Entity<TKey>
{
public Test()
{ }
public TestEntity TestMethod(TKey id)
{
List<TestEntity> testEntity = new List<TestEntity>();
testEntity.Add(new TestEntity(5));
return testEntity.SingleOrDefault(x => x.Id.Equals(id));
}
}
class Entity<TKey>
{
public TKey Id { get; set; }
}
class TestEntity : Entity
{
public TestEntity(int id)
{
Id = id;
}
}
class Entity : Entity<int>
{
}
它似乎工作得很好。所以今晚我会试试。
编辑#4
好的,我得到的异常是Canoot创建类型为“System.Object”的常量值。此上下文仅支持主要类型,如int32,string和guid。
答案 0 :(得分:2)
在您的存储库声明中
public abstract class RepositoryBase<TEntity, TKey> : IRepository<TEntity>, IDisposable
where TEntity : EntityBase<TKey>
where TKey : class
{
您已指定了只允许引用类型的类约束。参见
http://msdn.microsoft.com/en-us/library/d5x73970%28v=vs.80%29.aspx
其中T:class
type参数必须是引用类型,包括任何类, 接口,委托或数组类型。 (见下面的注释。)
删除:class约束以允许任何类型。
除非您参与学习练习,否则我不会尝试从头开始构建您的存储库。我会利用别人的所作所为。当我编写存储库框架时,我想要一个GetById方法,该方法可以使用不同类型的主键(尽管不是多列主键)。在撰写时,我发现以下两篇文章特别有用:
C# LINQ to SQL: Refactoring this Generic GetByID method
http://goneale.com/2009/07/27/linq-to-sql-generic-repository/
答案 1 :(得分:0)
它说Tkey必须是参考类型,但为什么?
你有约束:
where TKey : class
你需要约束吗?您想支持哪些关键类型?
答案 2 :(得分:0)
添加
where TKey : IEquatable<TKey>
应解决问题并允许您使用Equals语法。
答案 3 :(得分:0)
如果您使用f12键查看guid或int元数据,您将会实现它:
public struct int: IFormatable, IComparable, IComparable<int>, IEquatable<int>
public struct Guid: IFormatable, IComparable, IComparable<Guid>, IEquatable<Guid>
然后约束将是:
public interface IRepository<TEntity, TKey>
where TEntity: class, IEntity<TKey>
where TKey: struct, IFormatable, IComparable, IComparable<TKey>, IEquatable<TKey>
和IEntity接口必须是这样的:
public interface IEntity<TKey> where TKey: struct, IFormatable,
IComparable, IComparable<TKey>, IEquatable<TKey>
但是运算符==仍然不适合我,但是Equals方法按预期工作,仍然在寻找等于运算符的解决方案。