4

几周前,我加入了 MEF(ComponentModel)潮流,现在我将它用于我的很多插件和共享库。总的来说,除了我经常犯的错误,这导致调试会话令人沮丧之外,这一切都很好。

无论如何,我的应用程序运行良好,但与 MEF 相关的代码更改导致我的自动构建失败。我的大多数单元测试都失败了,仅仅是因为我正在测试的模块依赖于需要由 MEF 加载的其他模块。我通过绕过 MEF 并直接实例化这些对象来解决这些情况。

换句话说,通过 MEF 我会有类似的东西

[Import]
public ICandyInterface ci { get; set; }

[Export(typeof(ICandyInterface))]
public class MyCandy : ICandyInterface
{
    [ImportingConstructor]
    public MyCandy( [Import("name_param")] string name) {}
    ...
}

但在我的单元测试中,我只会使用

CandyInterface MyCandy = new CandyInterface( "Godiva");

此外,CandyInterface 需要连接到数据库,我只是通过将测试数据库添加到我的单元测试文件夹来解决这个问题,并且我让 NUnit 将它用于所有测试。

好的,所以这是我关于这种情况的问题:

  1. 这是做事的坏方法吗?
  2. 你会推荐在 [SetUp] 中组成部分吗?
  3. 我还没有学会如何在单元测试中使用模拟——这是我可能想模拟底层数据库连接(以某种方式)只返回虚拟数据而不真的需要数据库的一个很好的例子吗?
  4. 如果你以前遇到过这样的事情,你能提供你的经验和你解决问题的方法吗?(或者这应该进入社区维基吗?)
4

3 回答 3

10

听起来你在正确的轨道上。单元测试应该测试一个单元,这就是你直接创建实例时所做的。如果您让 MEF 为您编写实例,它们将倾向于集成测试。并不是说集成测试有什么问题,而是单元测试往往更易于维护,因为您单独测试每个单元。

您不需要容器来连接单元测试中的实例

我通常建议不要在 SetUp 中组合 Fixtures,因为它会导致General Fixture反模式。

最好用Test Doubles替换依赖项。动态模拟是执行此操作的更通用的方法之一,因此绝对是您应该学习的东西。

于 2010-05-18T07:39:51.237 回答
0

我同意手动创建 DOC 比使用 MEF 组合容器来满足导入要好得多,但是关于注释“在设置中组合固定装置会导致一般固定装置反模式” - 我想提一下,情况并非总是如此。

如果您使用静态容器并通过 CompositionInitializer.SatisfyImports 满足导入,您将不得不面对一般的夹具反模式,因为 CompositionInitializer.Initialize 不能被多次调用。但是,您始终可以在容器本身上创建 CompositionContainer、添加目录和调用 SatisyImportOnce。在这种情况下,您可以在每个测试中使用新的 CompositionContainer 并摆脱面对共享/通用夹具反模式

于 2010-11-03T15:28:06.870 回答
0

我在博客上写了如何使用 MEF 进行单元测试(不是 nunit,但工作方式相同)。诀窍是使用 MockExportProvider 并且我为我的所有测试创建了一个测试库来继承。

这是我的主要 AutoWire 功能,适用于集成和单元测试:

protected void AutoWire(MockExportProvider mocksProvider, params Assembly[] assemblies){

CompositionContainer container = null;

var assCatalogs = new List<AssemblyCatalog>();

foreach(var a in assemblies)
{
    assCatalogs.Add(new AssemblyCatalog(a));
}

if (mocksProvider != null)
{
    var providers = new List<ExportProvider>();

    providers.Add(mocksProvider); //need to use the mocks provider before the assembly ones            

    foreach (var ac in assCatalogs)
    {
        var assemblyProvider = new CatalogExportProvider(ac);                    
        providers.Add(assemblyProvider);
    }

    container = new CompositionContainer(providers.ToArray());

    foreach (var p in providers) //must set the source provider for CatalogExportProvider back to the container (kinda stupid but apparently no way around this)
    {
        if (p is CatalogExportProvider)
        {
            ((CatalogExportProvider)p).SourceProvider = container;
        }
    }
}
else
{
    container = new CompositionContainer(new AggregateCatalog(assCatalogs));
}

container.ComposeParts(this);        
}

关于我的帖子的更多信息:https ://yoavniran.wordpress.com/2012/10/18/unit-testing-wcf-and-mef/

于 2012-10-18T20:45:14.023 回答