DAL 和 BLL 的单元/集成测试(使用 lambda)

Posted

技术标签:

【中文标题】DAL 和 BLL 的单元/集成测试(使用 lambda)【英文标题】:Unit/Integration tests for DAL and BLL (with lambdas) 【发布时间】:2015-11-07 22:00:40 【问题描述】:

我的代码基本上是这样的:

数据访问合同:

public interface IProvideDataAccess<T>
    where T : Entity

    IEnumerable<T> Select(Func<T, bool> condition);
    void Save(T entity);
    void Delete(T entity);

数据访问层:

public class Db4oProvider<T> : IProvideDataAccess<T>
    where T : Entity

    private IEmbeddedConfiguration _configuration;
    private string _connectionString;

    public Db4oAccesDonnees(string connectionString)
    
        _connectionString = connectionString;
        _configuration = Db4oEmbedded.NewConfiguration();
    

    IEnumerable<T> IProvideDataAccess<T>.Select(Func<T, bool> condition)
    
        using (IObjectContainer db = Db4oEmbedded.OpenFile(_configuration, _connexion))
        
            return db.Query<T>(e => condition(e));
        
    
    void IProvideDataAccess<T>.Save(T entity)
    
        using (IObjectContainer db = Db4oEmbedded.OpenFile(_configuration, _connexion))
        
            db.Store(entity);
        
    
    void IProvideDataAccess<T>.Delete(T entity)
    
        using (IObjectContainer db = Db4oEmbedded.OpenFile(_configuration, _connexion))
        
            db.Delete(entity);
        
    

业务逻辑层:

public class MyRepository

    protected IProvideDataAccess<MyEntityType> _dataAccessProvider;

    public MyRepository(IProvideDataAccess<MyEntityType> dataAccessProvider)
    
        _dataAccessProvider = dataAccessProvider;
    

    public IEnumerable<MyEntityType> SelectValidEntities()
    
        return _dataAccessProvider.Select(e => /* test if entity is valid */);
    

单元/集成测试对我来说相当新,我不知道从哪里开始。

认为合乎逻辑的做法是

为 DAL 编写集成测试 为 BLL 编写单元测试(使用假 DAL)

我说的对吗?

我最大的问题是“选择”方法及其 Func 参数。我如何测试/模拟它?

基本上,我应该为这两个类写什么测试?

【问题讨论】:

【参考方案1】:

您的DAL 看起来像永远不会改变的类...在我看来,在它们上创建UTIT 是浪费时间。您应该使用良好的组件/“大型集成”(不仅是单元+数据库)测试来涵盖类的行为,例如Db4oProvider&lt;T&gt;。我建议您阅读有关Test Pyramid(In Martin Fowler blog 和this good article)的信息。我喜欢下面的金字塔:

关于Select 方法:

Integration test(unit + DB) 中,您必须将数据放入您的 DB 中(或保留一个带有数据的 DB...),使用条件调用 Select,然后验证您是否获得了预期的数据.

Component test 中,您需要将其作为被测行为的一部分进行测试,如果测试成功则一切正常...

UT 中,测试分为两种类型:

1.验证传递给Select 方法的条件(lambda)的行为的测试:(通过这种方式,您还可以验证Select 是否使用正确的参数调用...在我的示例中,我使用RhinoMocks)

var fakeProvider = MockRepository.GenerateStub<IProvideDataAccess<Entity>>();

fakeProvider.Select(o => o.Id =="asdfg");

fakeProvider.AssertWasCalled(access => access.Select(Arg<Func<Entity, bool>>
            .Matches(func => func(new Entity()
            
                Id = "asdfg"
            ))));

2.验证调用者方法行为的测试。在这些测试中,您应该配置您的 DAL 以返回一些重要的参数:

_selectResult = new List<Entity>();
fakeProvider.Stub(x => x.Select(Arg<Func<Entity, bool>>.Is.Anything))
            .Return(_selectResult);

【讨论】:

以上是关于DAL 和 BLL 的单元/集成测试(使用 lambda)的主要内容,如果未能解决你的问题,请参考以下文章

不同的 UI 共享相同的 BLL 和 DAL

如何在 UI、BLL、DAL 之间使用 DTO

DAL/BLL 和客户端/服务器:客户端应该使用 BLL 还是 DAL 对象进行演示?或者可能是另一层(数据传输对象?)

bll,dal 和接口实现

在我的 API 中使用 BLL 函数而不参考 DAL

DAL 和 BLL 应通过的类型