首页 > 解决方案 > AssemblyLoadContext,动态加载程序集,实例化对象并强制转换为共享接口

问题描述

我正在尝试使用AssemblyLoadContext(从 netcore 3.0 版中存在)加载程序集,实例化一个对象并将此对象转换为接口,但出现转换异常错误。

该接口在加载程序集的项目和实例化的实现之间共享。该对象显然已正确实例化,但是当我这样做时出现意外错误(T)instance

尝试使用观察者,我能够按照我正在使用的代码和观察者的屏幕截图将实例正确地投射到界面:

private (ExecutionAssemblyLoadContext, T) LoadTheAssemblyAndInstance<T>(string assemblyName, string typeNameToInstance)
{
    var basePath = Path.GetDirectoryName(Assembly.GetEntryAssembly().Location);
    var assemblyContext = new ExecutionAssemblyLoadContext($"{basePath}/{assemblyName}.dll");
    var assembly = assemblyContext.LoadFromAssemblyPath($"{basePath}/{assemblyName}.dll");
    var externalCodeEvent = typeNameToInstance != null ? assembly.ExportedTypes
        .Where(x => x.FullName == typeNameToInstance)
        .Single() : assembly.ExportedTypes.First();
    var instance = Activator.CreateInstance(
            externalCodeEvent,
            _defaultConstructorParameters
        );
    return (assemblyContext, (T)instance);
}

在此处输入图像描述

这是完整的异常消息:

System.InvalidCastException:'无法将'Expriva.NewWorkflow.BPMN.ExecutionCodeTest.ExecutionContractTest'类型的对象转换为'Expriva.NewWorkflow.ExternalShared.Interfaces.IExecutionContract'。

按照显示T由实例实现 的屏幕截图在此处输入图像描述

标签: c#.net.net-core.net-assembly.net-core-3.0

解决方案


AssemblyLoadContext 支持动态代码加载和卸载,它在自己的 AssemblyLoadContext 实例中创建用于加载代码及其依赖项的隔离上下文。

问题是在 ExecutionAssemblyLoadContext 实现中,依赖关系被解决和隔离。使用默认实现,AssemblyLoadContext 的文档表明共享类型不会被隔离。按照正确的实现来使用共享接口。

public class ExecutionAssemblyLoadContext : AssemblyLoadContext
{
    public ExecutionAssemblyLoadContext() : base(isCollectible: true)
    {
    }

    protected override Assembly Load(AssemblyName name)
    {
        return null;
    }
}

推荐阅读