• 2018-2-13-C#-动态加载卸载-DLL


    title author date CreateTime categories
    C# 动态加载卸载 DLL
    lindexi
    2018-2-13 17:23:3 +0800
    2018-2-13 17:23:3 +0800
    C#

    我最近做的软件,需要检测dll或exe是否混淆,需要反射获得类名,这时发现,C#可以加载DLL,但不能卸载DLL。于是在网上找到一个方法,可以动态加载DLL,不使用时可以卸载。

    我在写一个WPF 程序,发现可以通过 Assembly.Load 加载 DLL,但是如何卸载DLL?下面就来说下如何卸载。

    看到 Assembly.Load 是把 DLL 加载到当前程序集,这句话,我就想到了我们的主程序集和当前的不同,那么可以加载到当前不会影响主程序。那么如何新建一个程序集?他是否可以卸载,答案是可以的。

    首先,我们可以通过var appDomain = AppDomain.CreateDomain(appDomainName);创建 AppDomain 。他是可以卸载,卸载 AppDomain 使用 AppDomain.Unload ,就可以把加载在 AppDomain 的 DLL 卸载。

    于是我们需要把 DLL 加载在 AppDomain ,这样之后可以卸载 AppDomain 动态删掉 加载的DLL。

    如果要把 DLL 加载在 AppDomain 需要先写一个类,继承MarshalByRefObject

        internal class ApplicationProxy : MarshalByRefObject
        {
            public void DoSomething()
            {
                
            }
        }
    
        var proxy =
                    appDomain.CreateInstanceAndUnwrap(Assembly.GetAssembly(typeof(ApplicationProxy)).FullName,
                    typeof(ApplicationProxy).ToString()) as ApplicationProxy;

    我们可以在 DoSomething 函数加载 DLL ,加载的 DLL 在 AppDomain ,不在主程序,所以卸载 AppDomain 可以卸载 DLL

    假如是从 文件加载,可以使用 LoadFile

                      var assembly  = Assembly.LoadFile(file.FullName);

    assembly 可以获得所有的类和方法。

    然后需要卸载时,可以使用 AppDomain.Unload(appDomain);

    建议写var assembly = Assembly.LoadFile(file.FullName);在 try,写 AppDomain.Unload(appDomain);在 finally

    上面的 appDomainName 是我自己给他的。

    http://stackoverflow.com/questions/2132649/loading-unloading-assembly-in-different-appdomain

    我们可以验证,如果不使用新建一个 AppDomain 加载的 DLL 会在主程序集,如果使用了,就会在我们新建的 AppDomain 。

    首先我们使用 Assembly.LoadFile(file) 加载,再用反射获得当前程序集,然后获取他的所有 type ,当然我们是知道加载的 File 包含的 type,一会可以验证使用已经加载他。

                System.Reflection.Assembly.LoadFile(file);
    
                foreach (var assembly in AppDomain.CurrentDomain.GetAssemblies())
                {
                    //查看type               
                }

    可以看到 file 包含的 type 在主程序。

    我们使用新建 appDomain

                  const string appDomainName = "ConfuseChecker";
                var appDomain = AppDomain.CreateDomain(appDomainName);
                var proxy =
                           appDomain.CreateInstanceAndUnwrap(Assembly.GetAssembly(typeof(ApplicationProxy)).FullName,
                               typeof(ApplicationProxy).ToString()) as ApplicationProxy;
                proxy.DoSomething(new FileInfo(file));
    
                AppDomain.Unload(appDomain);

    这时可以看到,我们的主程序没有包含 file 的 type 。

  • 相关阅读:
    最小二乘法求回归直线方程的推导过程
    最小二乘法求回归直线方程的推导过程
    Redis过期键的删除策略
    Redis过期键的删除策略
    最小二乘法求回归直线方程的推导过程
    最小二乘法求回归直线方程的推导过程
    不用第三方实现内网穿透
    不用第三方实现内网穿透
    X Redo丢失的4种情况及处理方法
    Problem D: 逆置链式链表(线性表)
  • 原文地址:https://www.cnblogs.com/lindexi/p/12086535.html
Copyright © 2020-2023  润新知