1

我有一个 ITagger 和一个 IWpfTextViewMargin,它们都作为 MEF 组件导出。我想在我的 Margin 代码中导入 ITagger,然后在该 Tagger 中使用一些成员。

现在我尝试在 Margin 类中使用 ComponentContainer,然后导入 IViewTaggerProvider。我使用了以下代码,可以在许多 MEF 教程中找到

[Import(typeof(IViewTaggerProvider))]
public IViewTaggerProvider vt_provider { get; set; }

var catalog = new AggregateCatalog();
catalog.Catalogs.Add(new AssemblyCatalog(typeof(TestMargin).Assembly));
_container = new CompositionContainer(catalog);
//Fill the imports of this object
try
{
    this._container.ComposeParts(this);
}
catch (CompositionException compositionException)
{
    System.Diagnostics.Trace.WriteLine(compositionException.Message);
}

和出口代码。

[Export(typeof(IViewTaggerProvider))]
[ContentType...

导出的类在另一个命名空间中定义,但在同一个程序集中。

在这里,我遇到了 ComposeParts(this) 抛出 ImportCardinalityMismatchException 的问题。我不知道为什么参数是this。我试图将目录传递给它,没有例外,但导入也是空的。我还提到了调试 mef 失败,并认为导出的类具有正确的合同名称和导出类型标识。


在使用 Visual MEFx 检查程序集并调试后,我发现可能是因为 IViewTaggerProvider 导入了 Visual Studio IClassificationTypeRegistryService,它也是 MEF 部件,导致 IViewTaggerProvider 被拒绝。

[Primary Rejection]
 [Exception] System.ComponentModel.Composition.ImportCardinalityMismatchException: No valid exports were found that match the constraint '((exportDefinition.ContractName == "Microsoft.VisualStudio.Text.Classification.IClassificationTypeRegistryService") AndAlso (exportDefinition.Metadata.ContainsKey("ExportTypeIdentity") AndAlso "Microsoft.VisualStudio.Text.Classification.IClassificationTypeRegistryService".Equals(exportDefinition.Metadata.get_Item("ExportTypeIdentity"))))', invalid exports may have been rejected.

所以一种解决方案是添加导出 IClassificationTypeRegistryService 的程序集。这是一个 Visual Studio 核心编辑器服务,但我找不到哪个程序集导出它。有人知道吗?

还是有更好的可能解决方案?

4

1 回答 1

1

试试 VisualMEFx。这是一篇关于入门的简短博客文章https://ihadthisideaonce.com/2012/02/22/getting-started-with-visual-mefx/。启动并运行它后,使用 VisualMEFx 加载 TestMargin 程序集并查看是否从该程序集中导出了任何 IViewTaggerProvider。

还要记住 ImportCardinalityMistmatch 不仅意味着缺少导出。这也可能意味着有太多可用的导出可以满足导入,而 MEF 无法选择使用哪一个。因此,当您在 VisualMEFx 中检查您的合成时,请检查是否有太多。

这个参数:

void Bootstrap()
{
  var catalog = new AggregateCatalog();
  catalog.Catalogs.Add(new AssemblyCatalog(typeof(TestMargin).Assembly));
  _container = new CompositionContainer(catalog);

 //Fill the imports of this object
 try
 {
    var objectToSatisfy = this;
    // var objectToSatifsy = new SomeOtherObjectWithImports();

    this._container.ComposeParts(objectToSatisfy);
 }
 catch (CompositionException compositionException)
 {
    System.Diagnostics.Trace.WriteLine(compositionException.Message);
 }
}

当您调用时,ComposeParts您将一个对象传递给该方法。MEF 将获取您传递的对象,并查看其上是否有任何需要满足的导入。如果它找到任何导入,它将查看目录并尝试满足它们。您可以将所需的任何对象传递给该ComposeParts方法。因此,我稍微修改了您的示例代码以显示两个不同的选项。一种选择是创建一些需要满足的对象,然后将其交给容器进行组合。这就是我在注释掉的行中所做的var objectToSatisfy = new SomeOtherObjectWithImports()。但通常情况下,我们想要组合的对象与正在调用的对象相同ComposeParts. 所以我们不需要创建一个新对象来传递给容器,我们已经有了这个对象,我们只需要一个对它的引用。在 C# 中,我们可以使用关键字获取对当前对象实例的引用this(在 VB.NET 中,关键字是Me)。因此,当我们想要满足对调用 的同一个对象的导入时ComposeParts,我们可以通过使用this引用作为 的参数来实现ComposeParts

ComposeParts方法的参数是一个参数数组。非正式地,这只是意味着当你写container.ComposeParts(this)它时,它被解释为你写了container.ComposeParts(new object[] { this }). 实际上,这意味着您可以一次将多个对象传递给 MEF,如下所示:

container.ComposeParts(this, objectToSatifsy, thirdObjectToCompose);

如果调用的对象ComposeParts没有导入,那么您不应该将其this用作参数。相反,创建一个您想要组合的类型的对象并将其传递给方法。此外,除非您想要组合的所有零件在TestMargin装配体中都可用,否则您需要为确实提供零件的装配体创建更多 AssemlbyCatalog,并将它们添加到您的 AggregateCatalog。

于 2012-04-27T13:23:46.203 回答