gpt4 book ai didi

c# - 如何将MEF导入导出信息持久化到磁盘

转载 作者:太空狗 更新时间:2023-10-30 00:43:04 25 4
gpt4 key购买 nike

对于我的应用程序,如 this question 中所述我想使用 MEF 扫描可用的插件程序集,然后以序列化格式(例如一组字符串或内存流)存储所有可用的导入和导出信息。这是必要的,因为我需要通过 AppDomain 边界传输导入和导出信息而不加载插件程序集(本质上我想延迟加载插件)。我找到了一些引用资料,例如 this onethis one但是没有一个链接让我知道如何:

  • 从程序集中提取所有导入和导出
  • 序列化所有必需的导入/导出信息
  • 然后再将序列化信息重新组合成导入和导出。

我想我可以使用 ReflectionModelServices class创建导入/导出定义,但仍然留下序列化和反序列化部分。任何人都可以指出一些示例、文档或向我提供有关如何执行这些步骤的建议吗?

最佳答案

此问题的答案由 Kevin 在 MEF discussion list 上提供.事实证明,可以从 MEF 中提取所有需要的信息 ExportDefinitionImportDefinition具有以下代码片段的数据结构。

第一步是将装配类型加载到目录中。然后为目录中的每个部分迭代导入和导出定义。导出定义只能放在类型、方法、属性和字段上(我的代码目前忽略了这些)。因此,要处理导出,可以使用以下代码。

var exports = new List<Tuple<string, MemberInfo>>();
foreach (var export in part.ExportDefinitions)
{
var memberInfo = ReflectionModelServices.GetExportingMember(export);
Tuple<string, MemberInfo> exportDefinition = null;
switch (memberInfo.MemberType)
{
case MemberTypes.Method:
exportDefinition = new Tuple<string, MemberInfo>(export.ContractName, memberInfo.GetAccessors().First() as MethodInfo);
break;
case MemberTypes.NestedType:
case MemberTypes.TypeInfo:
exportDefinition = new Tuple<string, MemberInfo>(export.ContractName, memberInfo.GetAccessors().First() as Type);
break;
case MemberTypes.Property:
// this is a bit ugly because we assume that the underlying methods for a property are named as:
// get_PROPERTYNAME and set_PROPERTYNAME. In this case we assume that exports always
// have a get method.
var getMember = memberInfo.GetAccessors().Where(m => m.Name.Contains("get_")).First();
var name = getMember.Name.Substring("get_".Length);
var property = getMember.DeclaringType.GetProperty(name);
exportDefinition = new Tuple<string, MemberInfo>(export.ContractName, property);
break;

default:
throw new NotImplementedException();
}

exports.Add(exportDefinition);
}

为了处理只能放置在属性、参数和字段(再次被忽略)上的导入,可以使用以下代码:

public void ExtractImports()
{
var imports = new List<Tuple<string, string>>();
foreach (var import in part.ImportDefinitions)
{
SerializedImportDefinition importDefinition = !ReflectionModelServices.IsImportingParameter(import)
? importDefinition = CreatePropertyImport(import)
: importDefinition = CreateConstructorParameterImport(import);
}
}

private Tuple<string, string> CreatePropertyImport(ImportDefinition import)
{
var memberInfo = ReflectionModelServices.GetImportingMember(import);
if (memberInfo.MemberType != MemberTypes.Property)
{
throw new ArgumentOutOfRangeException("import");
}

// this is a bit ugly because we assume that the underlying methods for a property are named as:
// get_PROPERTYNAME and set_PROPERTYNAME. In this case we assume that imports always
// have a set method.
var getMember = memberInfo.GetAccessors().Where(m => m.Name.Contains("set_")).First();
var name = getMember.Name.Substring("set_".Length);
var property = getMember.DeclaringType.GetProperty(name);
return new Tuple<string, string>(import.ContractName, property.ToString());
}

private Tuple<string, string> CreateConstructorParameterImport(ImportDefinition import)
{
var parameterInfo = ReflectionModelServices.GetImportingParameter(import);
return new Tuple<string, string>(import.ContractName, parameterInfo.Value.ToString());
}

请注意,似乎无法通过方法参数提供导入,因此上面的代码不支持这些。

一旦处理完所有导出和导入,序列化存储的 MemberInfo 就是一个简单的问题。对象转换成字符串格式。

关于c# - 如何将MEF导入导出信息持久化到磁盘,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/12328243/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com