gpt4 book ai didi

c# - 在另一个域中创建实例和展开?

转载 作者:太空狗 更新时间:2023-10-29 18:14:54 25 4
gpt4 key购买 nike

出于某种原因(之前正在工作),我目前在使用 CreateInstanceAndUnwrap 时遇到问题。

我的流程是这样的:

我动态生成一些代码,它通过 MEF 从子目录加载 DLL。然后这些应用程序从这些 DLL 中加载不同的部分(按需)。我必须更新我的代码,现在包括一个包含调用程序集路径的 AppDomainSetup。

我正确地创建了新的 AppDomain -- 没有问题。当我尝试运行这段代码时:

object runtime = domain.CreateInstanceAndUnwrap(
typeof(CrossDomainApplication).Assembly.FullName,
typeof(CrossDomainApplication).FullName);

我有很多问题 -- 运行时(上面的变量)不再可以转换为 CrossDomainApplication 或 ICrossDomainApplication。

实际对象看起来像:

public class CrossDomainApplication : MarshalByRefObject, ICrossDomainApplication

界面看起来像这样:

public interface ICrossDomainApplication
{
void Run(CrossDomainApplicationParameters parameters);
}

参数如下:

[Serializable]
public class CrossDomainApplicationParameters : MarshalByRefObject
{
public object FactoryType { get; set; }
public Type ApplicationType { get; set; }
public string ModuleName { get; set; }
public object[] Parameters { get; set; }
}

运行时的 native 类型似乎是 MarshalByRefObject —— 它不喜欢转换为其他任何类型。

有什么想法是错误的吗?

编辑:这是我按以下方式运行时遇到的错误:

            ICrossDomainApplication runtime = (ICrossDomainApplication)domain.CreateInstanceAndUnwrap(
typeof(CrossDomainApplication).Assembly.FullName,
typeof(CrossDomainApplication).FullName);

//Exception before reaching here
runtime.Run(parameters);

System.InvalidCastException:无法将透明代理转换为类型“Infrastructure.ICrossDomainApplication”。

这是我创建域时的样子:

        AppDomain domain = AppDomain.CreateDomain(
Guid.NewGuid().ToString(),
null,
new AppDomainSetup()
{
ApplicationBase = GetPath(),
ConfigurationFile = AppDomain.CurrentDomain.SetupInformation.ConfigurationFile,
ApplicationName = AppDomain.CurrentDomain.SetupInformation.ApplicationName,
LoaderOptimization = LoaderOptimization.MultiDomainHost
});

和 GetPath() 看起来像这样:

    private string GetPath()
{
Uri path = new Uri(Assembly.GetCallingAssembly().CodeBase);

if (path.IsFile)
{
path = new Uri(path, Path.GetDirectoryName(path.AbsolutePath));
}

return path.LocalPath.Replace("%20", " ");
}

最佳答案

出于帮助其他穷人的愿望,在尝试了很多其他组合后,我终于弄明白了。

我不得不改变一些东西......首先是:

            AppDomain domain = AppDomain.CreateDomain(
Guid.NewGuid().ToString(),
AppDomain.CurrentDomain.Evidence,
new AppDomainSetup()
{
ApplicationBase = AppDomain.CurrentDomain.SetupInformation.ApplicationBase,
ConfigurationFile = AppDomain.CurrentDomain.SetupInformation.ConfigurationFile,
LoaderOptimization = LoaderOptimization.MultiDomainHost,
PrivateBinPath = GetPrivateBin(AppDomain.CurrentDomain.SetupInformation.ApplicationBase)
});

PrivateBinPath 是真正的技巧,它使其他一切(最终)开始工作。 PrivateBinPath(阅读文档)绝对需要是相对路径。如果您有一个名为“assemblies”的子文件夹,那么 PrivateBinPath 应该是“assemblies”。如果在它前面加上\或绝对路径,它将不起作用。

通过这样做,它导致 AssemblyResolve 事件最终触发。我通过以下方式做到了这一点(在创建新的子 AppDomain 之前):

            AppDomain.CurrentDomain.AssemblyResolve += ResolveAssembly;

ResolveAssembly 方法如下所示:

    private Assembly ResolveAssembly(object sender, ResolveEventArgs args)
{
var loadedAssemblies = AppDomain.CurrentDomain.GetAssemblies();

foreach (var assembly in loadedAssemblies)
{
if (assembly.FullName == args.Name)
{
return assembly;
}
}

return null;
}

使用 Linq 可以更轻松地编写该方法,但我保持这种方式是为了尝试让我自己明白为了调试目的而解析和加载的内容。

并且,一如既往,确保您断开事件(如果您一次加载 AppDomain):

            AppDomain.CurrentDomain.AssemblyResolve -= ResolveAssembly;

这解决了一切。感谢所有提供帮助的人!

关于c# - 在另一个域中创建实例和展开?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/24760543/

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