2

这是我的测试:

[TestMethod]
public void TestUnitOfWork()
{
    UnitOfWork unitOfWork = new UnitOfWork();

    unitOfWork.ContactRepository.Insert(new Contact
    {
        Id = Guid.NewGuid(),
        FirstName = "Dom",
        LastName = "A",
        Email = "dominicarchual@yahoo.com"
    });

    var contacts = unitOfWork.ContactRepository.Get(x => x.FirstName == "Dominic");

    Assert.AreEqual(1, contacts.Count());
}

我得到的错误是:

测试方法 MvcContacts.Tests.Controllers.HomeControllerTest.TestUnitOfWork 抛出异常:System.Data.ProviderIncompatibleException:从数据库获取提供程序信息时出错。这可能是由 Entity Framework 使用不正确的连接字符串引起的。检查内部异常以获取详细信息并确保连接字符串正确。---> System.Data.ProviderIncompatibleException:提供程序未返回 ProviderManifestToken 字符串。---> System.Data.SqlClient.SqlException:建立与 SQL Server 的连接时发生与网络相关或特定于实例的错误。服务器未找到或无法访问。验证实例名称是否正确以及 SQL Server 是否配置为允许远程连接。(提供者:SQL 网络接口,错误:

我没有设置任何数据库;即我的上下文如下所示:

namespace MvcContacts.DAL
{
    public class ContactsContext : DbContext
    {
        public DbSet<Contact> Contacts { get; set; }
    }
}

我不知道如何将它映射到我的数据库;但是,我认为我不必这样做,因为我只是想使用模拟数据进行测试。我错了吗?

E1:这是我的工作单元。

namespace MvcContacts.DAL
{
    public class UnitOfWork : IDisposable
    {
        private ContactsContext context = new ContactsContext();
        private GenericRepository<Contact> contactRepository;

        public GenericRepository<Contact> ContactRepository
        {
            get
            {
                if (this.contactRepository == null)
                {
                    this.contactRepository = new GenericRepository<Contact>(context);
                }
                return contactRepository;
            }
        }

        public void Save()
        {
            context.SaveChanges();
        }

        private bool disposed = false;

        protected virtual void Dispose(bool disposing)
        {
            if (!this.disposed)
            {
                if (disposing)
                {
                    context.Dispose();
                }
            }
            this.disposed = true;
        }

        public void Dispose()
        {
            Dispose(true);
            GC.SuppressFinalize(this);
        }
    }
}
4

1 回答 1

5

正如我所说,问题在于您实际上是在UnitOfWork. 我很确定,您的GenericRepository<>课程只是包含DbSet在您的上下文中。这是您创建“真实”数据库访问器的地方。

private ContactsContext context = new ContactsContext();

但问题是您误解了存储库的整个概念。工作单元是一些数据源的抽象。您不应该对抽象进行单元测试,而应该对依赖于它的某些功能进行单元测试。顺便说一句,DbContext根据该定义,它本身就是一个工作单元(来自 martinfowler.com):

维护受业务事务影响的对象列表,并协调更改的写入和并发问题的解决。

人们为什么不让它保持原样?因为它有一个缺陷。让我举例说明。好像您正在学习 ASP.Net MVC,所以让我们编写一些控制器:

public class ContactsController
{
    public ActionResult Index(int pageSize, int currentPage)
    {
         using(var db = new MvcLearningContext())
         {
             var contacts = db.Contacts
                              .Skip((currentPage - 1) * pageSize)
                              .Take(pageSize)
                              .ToList();
             return View(contacts);
         }
    }
}

您可能知道,MVC 的一大优势是能够对控制器逻辑进行单元测试。因此,让我们尝试编写一个简单的单元测试,以确保控制器操作不会返回超过给定页面大小的条目:

[TestMethod]
public void IndexShouldNotReturnMoreThanPageSizeResults()
{
    // arrange
    var controller = new ContactsController();

    // act
    var view = (ViewResult) controller.Index(10, 1);

    // assert  
    var Model = (IEnumerable<Contact>) view.Model;
    Assert.IsTrue(view.Model.Count() <= 10)
}

但是等等……我们不想在单元测试中查询真实的数据库。EF 的问题来了DbContext:它完全依赖于真实的数据库。但是我们怎样才能避免呢?UnitOfWork发挥作用:

public class ContactsController
{
    private UnitOfWorkFactoryBase _factory { get; set; }

    public ContactsController(UnitOfWorkFactoryBase factory)
    {
        factory = _factory;
    }

    public ActionResult Index(int pageSize, int currentPage)
    {
         using(var db = _factory.Create())
         {
             var contacts = db.Contacts
                              .Skip((currentPage - 1) * pageSize)
                              .Take(pageSize)
                              .ToList();
             return View(contacts);
         }
    }
}

单元测试代码:

[TestMethod]
public void IndexShouldNotReturnMoreThanPageSizeResults()
{
    // arrange
    var factory = new MockUnitOfWorkFactory();
    var controller = new ContactsController(factory);

    // act
    var view = (ViewResult) controller.Index(10, 1);

    // assert  
    var Model = (IEnumerable<Contact>) view.Model;
    Assert.IsTrue(view.Model.Count() <= 10)
}

在生产中你替换MockUnitOfWorkFactoryUnitOfWorkFactory

UPD:工厂的基本实现:

public abstract class UnitOfWorkFactoryBase
{
    public abstract UnitOfWorkBase Create();
}

public class UnitOfWorkFactory : UnitOfWorkFactoryBase
{
    public override UnitOfWorkBase Create()
    {
        return new UnitOfWork();
    }
}

public class MockUnitOfWorkFactory : UnitOfWorkFactoryBase
{
    public override UnitOfWorkBase Create()
    {
        return new MockUnitOfWork();
    }
}

UnitOfWork并且MockUnitOfWork是 UnitOfWorkBase 抽象类的实现。

于 2013-09-17T15:35:42.627 回答