在我正在进行的项目中,我们有许多只通过反射实例化的类,它们所属的程序集不会被任何其他类引用。
此方案允许在运行时替换程序集以扩展/更正前一个程序集的功能。请注意,在我们的特定情况下,不可能使用MarshalByRefObject,因此同一程序集的不同版本,暴露具有相同名称但功能不同的类,最终会加载到同一个AppDomain中。
我提供了一个极其简化的示例:考虑以下接口声明和控制台应用程序
public interface ISomeInterface
{
string getData();
}
class Program
{
static void Main(string[] args)
{
byte[] lastLibrary = null;
Assembly lastAssembly = null;
while (true)
{
byte[] theLibrary = File.ReadAllBytes("ClassLibrary1.dll");
if (lastLibrary == null || !theLibrary.SequenceEqual(lastLibrary))
{
lastAssembly = Assembly.Load(theLibrary);
}
ISomeInterface obj = lastAssembly.CreateInstance("ClassLibrary1.Class1") as ISomeInterface;
Console.WriteLine(obj.getData());
lastLibrary = theLibrary;
Thread.Sleep(1000);
}
}
}
以及以下接口实现(在单独的程序集中):
public class Class1 : ISomeInterface
{
public string getData()
{
return "someText";
}
}
正如预期的那样,如果我更改了ISomeInterface的实现(例如,将“someText”更改为“someOtherText”)并将旧的dll替换为新的dll,则使用最新版本。但我想知道:这种机制会给软件的稳定性带来问题的可能性是多少?我的意思是,在我通过反射实例化这些类并且没有其他组件引用它们之前,一切都是安全的吗?至于我们的初步测试,看起来一切正常,但我也希望听到专家的意见。
我知道在这个具体的例子中,可以使用MarshalByRefObject来完成所有事情,但正如我之前所说,我们不可能朝那个方向前进。
答案 0 :(得分:4)
这样的事应该可以正常工作,但我认为你误解了实际发生的事情。
将程序集加载到AppDomain时,只要该AppDomin存在,就会从不发布。你要做的是加载一个新的程序集。但通过这样做,你不发布旧版本。例如,如果来自先前版本的对象仍然存在,则它将返回“someText”,即使新创建的对象将返回“someOtherText”。虽然这两个对象的类型名称都是ClassLibrary1.Class1
,但它们是不同的类型。
如果这种行为对你来说是正确的,那么你应该没问题。但是不要忘记所有旧版本仍然被加载并因此保存在内存中,因此这样做实际上会导致内存泄漏。