0

我正在创建一个模拟 IDbSet 以允许对实体框架类进行单元测试等。

但是我真的很难检测到变化,甚至根本不知道如何去做。到目前为止,这是我的课程...

public interface IReportContext
{
    IDbSet<Report> Reports {get;}
    public int SaveChanges();
}

public class MockReportContext : IReportContext
{
    IDbSet<Report> Reports {get;}

    public int SaveChanges()
    {
        //Need to detect changes here???
    }

    public MockReportContext()
    {
       Reports = new MockDbSet<Report>();
    }
}

public class MockDbSet<T> : IDbSet<T>
{
    readonly ObservableCollection<T> _data;
    readonly IQueryable _query;

    public FakeDbSet()
    {
        _data = new ObservableCollection<T>();
        _query = _data.AsQueryable();
    }

    public FakeDbSet(ObservableCollection<T> data)
    {
        _data = data;
        _query = _data.AsQueryable();
    }

    public virtual T Find(params object[] keyValues)
    {
        throw new NotImplementedException();
    }

    public T Add(T item)
    {
        _data.Add(item);
        return item;
    }

    public T Remove(T item)
    {
        _data.Remove(item);
        return item;
    }

    public T Attach(T item)
    {
        _data.Add(item);
        return item;
    }

    public T Detach(T item)
    {
        _data.Remove(item);
        return item;
    }

    public T Create()
    {
        return Activator.CreateInstance<T>();
    }

    public TDerivedEntity Create<TDerivedEntity>() where TDerivedEntity : class, T
    {
        return Activator.CreateInstance<TDerivedEntity>();
    }

    public ObservableCollection<T> Local
    {
        get { return _data; }
    }

    Type IQueryable.ElementType
    {
        get { return _query.ElementType; }
    }

    System.Linq.Expressions.Expression IQueryable.Expression
    {
        get { return _query.Expression; }
    }

    IQueryProvider IQueryable.Provider
    {
        get { return _query.Provider; }
    }

    System.Collections.IEnumerator System.Collections.IEnumerable.GetEnumerator()
    {
        return _data.GetEnumerator();
    }

    IEnumerator<T> IEnumerable<T>.GetEnumerator()
    {
        return _data.GetEnumerator();
    }
}

这适用于添加、删除和检索实体。但是,当我尝试以下操作时:

IReportContext context = new MockReportContext();
context.Reports.Add(new Report()); //Works
Report report = context.Reports.First(); //Works
report.Message = "Hello World!";
context.SaveChanges(); //Does nothing

MockReportContext 怎么知道它返回的报表对象已经改变了?我知道使用实体框架可以做到这一点,所以它一定是可能的,但我不知道如何......

4

1 回答 1

0

我想你大部分时间都在那里,但我建议使用像Moq(我个人偏好)或Rhino Mocks这样的模拟框架来模拟IReportContext你的单元测试,而不是去创建一个像. (好吧,学习一个 mocking 框架确实会有些麻烦,但它会节省很多 Fake 类的苦差事。)MockReportContext

我假设您正在对依赖于 的代码进行单元测试IReportContext,因此您不需要在内部做任何事情SaveChanges(),您只需要断言您的代码确实在SaveChanges()内部调用(如果应该)。

这是一个很好的概述,将 Moq 与 Entity Framework 的派生 DbContext / DbSet 类一起使用。

在上面链接的概述中,如果最后的单元测试正在测试内部调用的方法SaveChanges(),它还可以通过以下行验证SaveChanges()您的方法确实调用了该方法:

dc.Verify(db => db.SaveChanges());

您也可以MockReportContext通过在类中将 side 属性设置为 TrueSaveChanges()并在单元测试结束时对其进行检查来对您的类执行此操作,但是模拟框架更加灵活,并且无需为单元测试编写额外的类。

如果您想避免使用模拟框架,那么这里是使用 fakes的方法。

于 2014-03-17T19:38:11.120 回答