存储库模式和内存中的单元测试

时间:2013-07-27 21:23:33

标签: c# .net unit-testing repository-pattern

我已经看到了Repository Pattern的一些实现,非常简单直观,在stackoverflow中链接形式的其他答案

http://www.codeproject.com/Tips/309753/Repository-Pattern-with-Entity-Framework-4-1-and-C http://www.remondo.net/repository-pattern-example-csharp/

public interface IRepository<T>
{
    void Insert(T entity);
    void Delete(T entity);
    IQueryable<T> SearchFor(Expression<Func<T, bool>> predicate);
    IQueryable<T> GetAll();
    T GetById(int id);
}

public class Repository<T> : IRepository<T> where T : class, IEntity
{
    protected Table<T> DataTable;

    public Repository(DataContext dataContext)
    {
        DataTable = dataContext.GetTable<T>();
    }
...

如何在进行单元测试时将其设置为在内存中工作?有没有办法从内存中的任何东西构建DataContext或Linq表?我的想法是创建一个集合(List,Dictionary ...)并在单元测试时将其存根。

谢谢!

编辑: 我需要这样的东西:

  • 我有一本课本
  • 我有一个类库
  • Library构造函数中,我初始化了存储库:

    var bookRepository = new Repository<Book>(dataContext)

  • Library方法使用存储库,就像这样

    public Book GetByID(int bookID)
    { 
        return bookRepository.GetByID(bookID)
    }
    

测试时,我想提供一个内存上下文。在生产中,我将提供真实的数据库上下文。

1 个答案:

答案 0 :(得分:19)

我建议使用像MoqRhinoMocks这样的模拟库。使用Moq的一个很好的教程可以找到here

在您决定使用哪一个之前,以下内容可能有所帮助:

其他信息:可以找到单元测试框架的比较here


OP请求后

更新

创建内存数据库

var bookInMemoryDatabase = new List<Book>
{
    new Book() {Id = 1, Name = "Book1"},
    new Book() {Id = 2, Name = "Book2"},
    new Book() {Id = 3, Name = "Book3"}
};

模拟您的存储库(我使用Moq作为以下示例)

var repository = new Mock<IRepository<Book>>();

设置存储库

// When I call GetById method defined in my IRepository contract, the moq will try to find
// matching element in my memory database and return it.

repository.Setup(x => x.GetById(It.IsAny<int>()))
          .Returns((int i) => bookInMemoryDatabase.Single(bo => bo.Id == i));

通过在构造函数参数

中传递模拟对象来创建库对象
var library = new Library(repository.Object);

最后进行了一些测试:

// First scenario look up for some book that really exists 
var bookThatExists = library.GetByID(3);
Assert.IsNotNull(bookThatExists);
Assert.AreEqual(bookThatExists.Id, 3);
Assert.AreEqual(bookThatExists.Name, "Book3");

// Second scenario look for some book that does not exist 
//(I don't have any book in my memory database with Id = 5 

Assert.That(() => library.GetByID(5),
                   Throws.Exception
                         .TypeOf<InvalidOperationException>());

// Add more test case depending on your business context
.....