我尝试使用
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
是由实例实现的
来自 Eric Ouellet,而不是问题的作者:我已经做了一个示例并将其放在 Github 中,以便更容易地查看和测试问题所在。该代码可在以下位置访问:AssemblyLoadContext。
我也遇到了这个问题,将其视为不同的 AssemblyLoadContext 拥有自己的加载类型副本,因此尽管您可以在两个上下文中引用共享程序集,但就它们而言,所包含的类型是唯一的。
解决方案是使用派生 ALC 中的加载覆盖来解析共享上下文中的程序集。
例如,如果共享类型加载到默认 ALC 中,这很简单。
public class ModAssemblyLoadContext : AssemblyLoadContext
{
public ModAssemblyLoadContext()
: base("ModAssemblyLoadContext", isCollectible: true)
{
}
protected override Assembly Load(AssemblyName assemblyName)
{
return Default.Assemblies
.FirstOrDefault(x => x.FullName == assemblyName.FullName);
}
}
AssemblyLoadContext 支持动态代码加载和卸载,它创建一个隔离的上下文,用于在自己的 AssemblyLoadContext 实例中加载代码及其依赖项。
问题在于,在 ExecutionAssemblyLoadContext 实现中,依赖关系得到了解决和隔离。使用默认实现(由 AssemblyLoadContext 文档表明),共享类型将不会被隔离。按照正确的实现来使用共享接口。
public class ExecutionAssemblyLoadContext : AssemblyLoadContext
{
public ExecutionAssemblyLoadContext() : base(isCollectible: true)
{
}
protected override Assembly Load(AssemblyName name)
{
return null;
}
}