当我们发布系统时,有时候希望不用关掉应用程序就能完成发布,但Spring.NET的ApplicationContext是从AppDomain.CurrentDomain中加载的程序集中创建对象的,并不支持从动态加载的程序集中创建对象,如果直接把更新后的程序集复制到bin目录,会无法替换dll或导致应用程序重启。最近我正好有这个需求,就研究了一下Spring的相关代码,需要解决的问题如下:

1.首先要解决如何动态加载程序集

2.其次要找到某种方式告诉Spring在创建对象的时候用我们自己加载进来的程序集

如何动态加载程序集

     动态加载程序集,网上最多的说法是另外创建一个AppDomain,然后在新创建的AppDomain里加载程序集。但这种方式不太不适合我,因为要想让一个对象能够穿过AppDomain边界,必须要继承MarshalByRefObject类,否则无法被其他AppDomain使用,而且从一个AppDomain调用另外一个AppDomain里的程序集时,需要多一层通信,这样太复杂了,而且出了问题也不好调试。

     我的主程序,利用Spring的IOC容器取出对象,然后调用对象的方法,这种应用场景,最适合的还是只存在一个AppDomain,所以得考虑另外的方法来动态加载程序集。

     动态加载程序集的目的,就是为了可以在不关闭应用程序的情况下替换dll,如果我直接Assembly.LoadFile肯定是不行的,因为我一旦Load了这个File,这个文件就被使用了,没法替换。

     所以首先要把程序集复制到临时目录,然后用Assembly.LoadFile去加载临时目录中的程序集,这样就可以在运行期替换程序集了。当然,我们还需要一个FileSystemWatcher来监控程序集的目录,当目录中的程序集发生变化时,再把新的程序集复制到新的临时目录,然后再加载新临时目录中的程序集文件。

     当然,已经加载的程序集在AppDomain.CurrentDomain没有被销毁前,是不能卸载的,所以经过几次程序集的更新后,我们的AppDomain.CurrentDomain中就会存在几个不同版本的程序集。为了区分出哪个程序集才是最新的,我们还需要一个全局的dictionary来存放最新的程序集,这个dictionary以程序集的名字作为key,以最新的加载后的程序集作为value。每次加载完临时目录的程序集之后,要更新这个dictionary。

     相关代码我在这里就不贴了,大家可以查看附件里的源代码。需要注意的是由于FileSystemWatcher在文件被修改时会多次触发Changed事件,所以为了避免多次加载同一个程序集,我稍微处理了一下,加了一个Timer,当定时器事件触发的时候才去加载程序集,而不是文件一被修改了就去加载。

如何让Spring用我们的程序集创建对象

     Spring在创建对象的时候,是利用反射,根据type的字符串形式来加载System.Type类型。它的ResolveType的核心逻辑如下(位于Spring.Core.TypeResolution.TypeResolver中):

public virtual Type Resolve(string typeName)
{
    if (StringUtils.IsNullOrEmpty(typeName))
    {
        throw BuildTypeLoadException(typeName);
    }
    TypeAssemblyHolder typeInfo = new TypeAssemblyHolder(typeName);
    Type type = null;
    try
    {
        type = typeInfo.IsAssemblyQualified ? LoadTypeDirectlyFromAssembly(typeInfo) : LoadTypeByIteratingOverAllLoadedAssemblies(typeInfo);
    }
    catch (Exception exception)
    {
        if (exception is TypeLoadException)
        {
            throw;
        }
        throw BuildTypeLoadException(typeName, exception);
    }
    if (type == null)
    {
        throw BuildTypeLoadException(typeName);
    }
    return type;
}

相关文章: