我正在尝试执行以下操作。应用程序A是"母亲应用程序"。它保持开放。 App B只是一个.DLL,我写了一些从App A中指定的接口派生的类。
然后,从App A,我将" import"来自App B的类并在其中运行方法。我希望能够动态更改App B(更改代码并重新编译)并在App A中使用新代码。
我在App B中有一个后编译命令,它将新的.DLL复制到App A目录。 App A创建一个新的AppDomain并使用ShadowCopying。我认为这就足够了,但是当我尝试重新编译时复制App B的新.DLL,它表示该文件正在使用中,无法覆盖。
以下是我现在的代码:
应用A(代码中的TestServer):
namespace TestServer
{
public interface IRunnable
{
void Run();
}
class Program
{
static void Main(string[] args)
{
AppDomainSetup setup = new AppDomainSetup();
setup.ApplicationName = "DemoApp";
setup.ApplicationBase = Environment.CurrentDirectory;
setup.ShadowCopyDirectories = Environment.CurrentDirectory;
setup.ShadowCopyFiles = "true";
int _domain = 1;
while (true)
{
string typeName = Console.ReadLine();
AppDomain appDomain = AppDomain.CreateDomain("DemoDomain" + _domain, null, setup);
IRunnable runner = appDomain.CreateInstanceFromAndUnwrap("TestClient.dll", typeName) as IRunnable;
runner.Run();
AppDomain.Unload(appDomain);
_domain++;
}
}
}
}
App B(代码中的TestClient):
namespace TestClient
{
public class TestRunner : TestServer.IRunnable
{
public void Run()
{
Console.WriteLine("RUNNING");
}
}
public class Bob : TestServer.IRunnable
{
public void Run()
{
Console.WriteLine("BOB");
}
}
}
我已经读过,如果您使用来自其他应用域的内容,那些应用域可以自动加载.DLL或其他内容。在这种情况下,我担心使用接口会导致基本AppDomain加载.DLL,从而将其锁定。
如何解决此问题?是否有更好的设置?
注意:我已经更新了我的代码,但仍会产生相同的结果。
答案 0 :(得分:3)
您的代码仍然在母AppDomain中运行,因为您将程序集和类型引入其中。任何代码都应该在衍生域中运行。我已经展示了一种在我的网站上设置类似内容的方法:A simple way to start your code in a different AppDomain
我不是百分百肯定,但这肯定是你必须采取的一步
<强>更新强>
就那里提出的解决方案而言,运行器的实例化将发生在DomainLifetimeHook的继承者中。显示的基础结构确保Start和Stop方法在AppDomainExpander类创建的AppDomain中运行。 Expander是创建新域的类,因此您的域设置应该放在域的Create方法中。
答案 1 :(得分:0)
主要问题在于:
Type type = assm.GetType("TestClient." + typeName);
这发生在App A的主AppDomain中,结果是主AppDomain锁定App B的程序集.dll
flq对他博客文章的回答中的链接应该适合你。
答案 2 :(得分:0)
只要打开ObjectHandle,就必须将类型加载到主appdomain中。要正常工作,您需要对未解包的ObjectHandle进行操作。