我想用委托(delegate)在单独的 AppDomain 中执行一些代码。我该怎么做?
UPD1:关于我的问题的更多细节 我的程序处理一些数据(一个迭代是:从数据库获取一些数据,评估它并在运行时创建程序集,执行动态程序集并将结果写入数据库)。
当前解决方案:每次迭代都在单独的线程中运行。 更好的解决方案:每次迭代都在单独的 AppDomain 中运行(以卸载动态组件)。
UPD2:所有,感谢您的回答。
我在这个线程中为我找到了一个: Replacing Process.Start with AppDomains
最佳答案
虽然您可以调用将由单独的 AppDomain 处理的委托(delegate),但我个人一直使用“CreateInstanceAndUnwrap”方法,该方法在外部应用程序域中创建一个对象并返回一个代理给它。
为此,您的对象必须继承自 MarshalByRefObject。
这是一个例子:
public interface IRuntime
{
bool Run(RuntimesetupInfo setupInfo);
}
// The runtime class derives from MarshalByRefObject, so that a proxy can be returned
// across an AppDomain boundary.
public class Runtime : MarshalByRefObject, IRuntime
{
public bool Run(RuntimeSetupInfo setupInfo)
{
// your code here
}
}
// Sample code follows here to create the appdomain, set startup params
// for the appdomain, create an object in it, and execute a method
try
{
// Construct and initialize settings for a second AppDomain.
AppDomainSetup domainSetup = new AppDomainSetup()
{
ApplicationBase = AppDomain.CurrentDomain.SetupInformation.ApplicationBase,
ConfigurationFile = AppDomain.CurrentDomain.SetupInformation.ConfigurationFile,
ApplicationName = AppDomain.CurrentDomain.SetupInformation.ApplicationName,
LoaderOptimization = LoaderOptimization.MultiDomainHost
};
// Create the child AppDomain used for the service tool at runtime.
childDomain = AppDomain.CreateDomain(
"Your Child AppDomain", null, domainSetup);
// Create an instance of the runtime in the second AppDomain.
// A proxy to the object is returned.
IRuntime runtime = (IRuntime)childDomain.CreateInstanceAndUnwrap(
typeof(Runtime).Assembly.FullName, typeof(Runtime).FullName);
// start the runtime. call will marshal into the child runtime appdomain
return runtime.Run(setupInfo);
}
finally
{
// runtime has exited, finish off by unloading the runtime appdomain
if(childDomain != null) AppDomain.Unload(childDomain);
}
在上面的示例中,它被编码为执行一个传递一些设置信息的'Run'方法,并且确定Run方法的完成表明子AppDomain中的所有代码都已完成运行,所以我们有一个finally确保卸载 AppDomain 的 block 。
您通常可能需要小心将哪些类型放置在哪些程序集中 - 您可能需要使用一个接口(interface)并将其放置在一个单独的程序集中,调用者(我们设置应用程序域的代码,并调用它) 和实现者(运行时类)是依赖的。此 IIRC 允许父 AppDomain 仅加载包含接口(interface)的程序集,而子应用程序域将加载包含运行时的程序集及其依赖项(IRuntime 程序集)。 IRuntime 接口(interface)(例如我们的 RuntimeSetupInfo 类)使用的任何用户定义的类型通常也应该放在与 IRuntime 相同的程序集中。此外,请注意如何定义这些用户定义的类型——如果它们是数据传输对象(如 RuntimeSetupInfo 可能是),你可能应该用 [serializable] 属性标记它们——以便传递对象的副本(序列化自 child 的父应用程序域)。您希望避免将调用从一个应用程序域编码到另一个应用程序域,因为这非常慢。按值传递 DTO(序列化)意味着访问 DTO 上的值不会引起跨单元调用(因为子应用程序域有它自己的原始副本)。当然,这也意味着值的变化并没有反射(reflect)在父应用域的原始DTO中。
如示例中的代码所示,父应用程序域实际上最终将同时加载 IRuntime 和 Runtime 程序集,但那是因为在对 CreateInstanceAndUnwrap 的调用中,我使用 typeof(Runtime) 来获取程序集名称和完全限定的类型名称.您可以改用硬编码或从文件中检索这些字符串 - 这将解耦依赖性。
AppDomain 上还有一个名为“DoCallBack”的方法,看起来它允许调用外部 AppDomain 中的委托(delegate)。但是,它采用的委托(delegate)类型是“CrossAppDomainDelegate”类型。其中的定义是:
public delegate void CrossAppDomainDelegate()
因此,它不允许您向其中传递任何数据。而且,由于我从未使用过它,所以我无法告诉您是否有任何特殊问题。
此外,我建议查看 LoaderOptimization 属性。您将其设置为什么会对性能产生重大影响,因为此属性的某些设置会强制新的应用程序域加载所有程序集的单独副本(以及 JIT 等),即使(IIRC)程序集位于 GAC 中(即这包括 CLR 程序集)。如果您使用子应用程序域中的大量程序集,这会给您带来可怕的性能。例如,我使用了来自子应用程序域的 WPF,这导致我的应用程序出现巨大的启动延迟,直到我设置了更合适的加载策略。
关于c# - 在单独的 AppDomain 中传递和执行委托(delegate),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/2008691/
我在使用omniauth/openid时遇到了一些麻烦。在尝试进行身份验证时,我在日志中发现了这一点:OpenID::FetchingError:Errorfetchinghttps://www.google.com/accounts/o8/.well-known/host-meta?hd=profiles.google.com%2Fmy_username:undefinedmethod`io'fornil:NilClass重要的是undefinedmethodio'fornil:NilClass来自openid/fetchers.rb,在下面的代码片段中:moduleNetclass
我遵循了教程http://gettingstartedwithchef.com/,第1章。我的运行list是"run_list":["recipe[apt]","recipe[phpap]"]我的phpapRecipe默认Recipeinclude_recipe"apache2"include_recipe"build-essential"include_recipe"openssl"include_recipe"mysql::client"include_recipe"mysql::server"include_recipe"php"include_recipe"php::modul
我在用Ruby执行简单任务时遇到了一件奇怪的事情。我只想用每个方法迭代字母表,但迭代在执行中先进行:alfawit=("a".."z")puts"That'sanalphabet:\n\n#{alfawit.each{|litera|putslitera}}"这段代码的结果是:(缩写)abc⋮xyzThat'sanalphabet:a..z知道为什么它会这样工作或者我做错了什么吗?提前致谢。 最佳答案 因为您的each调用被插入到在固定字符串之前执行的字符串文字中。此外,each返回一个Enumerable,实际上您甚至打印它。试试
如何在ruby中调用C#dll? 最佳答案 我能想到几种可能性:为您的DLL编写(或找人编写)一个COM包装器,如果它还没有,则使用Ruby的WIN32OLE库来调用它;看看RubyCLR,其中一位作者是JohnLam,他继续在Microsoft从事IronRuby方面的工作。(估计不会再维护了,可能不支持.Net2.0以上的版本);正如其他地方已经提到的,看看使用IronRuby,如果这是您的技术选择。有一个主题是here.请注意,最后一篇文章实际上来自JohnLam(看起来像是2009年3月),他似乎很自在地断言RubyCL
我正在尝试在Ruby中复制Convert.ToBase64String()行为。这是我的C#代码:varsha1=newSHA1CryptoServiceProvider();varpasswordBytes=Encoding.UTF8.GetBytes("password");varpasswordHash=sha1.ComputeHash(passwordBytes);returnConvert.ToBase64String(passwordHash);//returns"W6ph5Mm5Pz8GgiULbPgzG37mj9g="当我在Ruby中尝试同样的事情时,我得到了相同sha
我没有找到太多关于如何执行此操作的信息,尽管有很多关于如何使用像这样的redirect_to将参数传递给重定向的建议:action=>'something',:controller=>'something'在我的应用程序中,我在路由文件中有以下内容match'profile'=>'User#show'我的表演Action是这样的defshow@user=User.find(params[:user])@title=@user.first_nameend重定向发生在同一个用户Controller中,就像这样defregister@title="Registration"@user=Use
我正在使用RubyonRails3.0.9,我想生成一个传递一些自定义参数的link_toURL。也就是说,有一个articles_path(www.my_web_site_name.com/articles)我想生成如下内容:link_to'Samplelinktitle',...#HereIshouldimplementthecode#=>'http://www.my_web_site_name.com/articles?param1=value1¶m2=value2&...我如何编写link_to语句“alàRubyonRailsWay”以实现该目的?如果我想通过传递一些
如何检查Ruby文件是否是通过“require”或“load”导入的,而不是简单地从命令行执行的?例如:foo.rb的内容:puts"Hello"bar.rb的内容require'foo'输出:$./foo.rbHello$./bar.rbHello基本上,我想调用bar.rb以不执行puts调用。 最佳答案 将foo.rb改为:if__FILE__==$0puts"Hello"end检查__FILE__-当前ruby文件的名称-与$0-正在运行的脚本的名称。 关于ruby-检查是否
如何在Ruby中按名称传递函数?(我使用Ruby才几个小时,所以我还在想办法。)nums=[1,2,3,4]#Thisworks,butismoreverbosethanI'dlikenums.eachdo|i|putsiend#InJS,Icouldjustdosomethinglike:#nums.forEach(console.log)#InF#,itwouldbesomethinglike:#List.iternums(printf"%A")#InRuby,IwishIcoulddosomethinglike:nums.eachputs在Ruby中能不能做到类似的简洁?我可以只
C#实现简易绘图工具一.引言实验目的:通过制作窗体应用程序(C#画图软件),熟悉基本的窗体设计过程以及控件设计,事件处理等,熟悉使用C#的winform窗体进行绘图的基本步骤,对于面向对象编程有更加深刻的体会.Tutorial任务设计一个具有基本功能的画图软件**·包括简单的新建文件,保存,重新绘图等功能**·实现一些基本图形的绘制,包括铅笔和基本形状等,学习橡皮工具的创建**·设计一个合理舒适的UI界面**注明:你可能需要先了解一些关于winform窗体应用程序绘图的基本知识,以及关于GDI+类和结构的知识二.实验环境Windows系统下的visualstudio2017C#窗体应用程序三.