.Net Core 动态加载和卸载程序集

从 .Net Core 3.0开始支持程序集的加载和卸载,在 .Net FrameWork中使用独立的应用程序域来实现同样的功能,.Net Core 不支持创建多个应用程序域,所以无法使用多个应用程序域来实现程序集动态加载和卸载。

AssemblyLoadContext

程序集加载上下文,它代表一个程序集加载域,在这个域中可以加载、解析、卸载程序集

创建一个 .Net 库项目

cs 复制代码
    public class Class1
    {
        public void Fn1()
        {
            Console.WriteLine("Fn1");
        }

        public void Fn2()
        {
            Console.WriteLine("Fn2");
        }
    }

创建一个控制台项目

实现一个自定义程序集加载上下文类, isCollectible 表示是否支持程序集卸载,默认是 false,所以需要从 AssemblyLoadContext 继承并实现一个AssemblyLoadContext。

cs 复制代码
    internal class MyAssemblyContext:AssemblyLoadContext
    {
        public MyAssemblyContext() : base(isCollectible: true)
        { }
    }

测试加载和卸载方法

cs 复制代码
        //告诉编译器不要内联该方法,防止栈上存在引用导致无法UnLoad。
        [MethodImpl(MethodImplOptions.NoInlining)]
        static void CallLib(string assemblyPath, out WeakReference alcWeakRef)
        {
            MyAssemblyContext mlc = new MyAssemblyContext();

            //加载 assemblyPath 程序集到 MyAssemblyContext 上下文中
            Assembly clib = mlc.LoadFromAssemblyPath(assemblyPath);

            //引用 程序集加载上下文,用于确认当UnLoad时,上下文被正确清理掉了。
            alcWeakRef = new WeakReference(mlc, trackResurrection: true);

            Console.WriteLine("Load ClassLibrary1.dll success");

            Type t= clib.GetType("ClassLibrary1.Class1");
            object obj= Activator.CreateInstance(t);
            t.GetMethod("Fn1").Invoke(obj, new object[] { });

            Console.ReadKey();

            //清理程序集加载上下文并开始卸载该上下文中加载的程序集
            mlc.Unload();

            Console.WriteLine("UnLoad ClassLibrary1.dll success");
        }

测试程序集动态加载和卸载

cs 复制代码
            WeakReference testAlcWeakRef;
            CallLib("E:\\projs\\analyze_dump\\example_net\\net_demo\\ClassLibrary1\\bin\\Debug\\net7.0\\ClassLibrary1.dll",out testAlcWeakRef);

            //显示调用GC清理程序集加载上下文。
            for (int i = 0; testAlcWeakRef.IsAlive && (i < 10); i++)
            {
                GC.Collect();
                GC.WaitForPendingFinalizers();
            }

            Console.ReadKey();

成功加载程序集并实例化程序集中 Class1类并调用Fn1方法

在未卸载程序集前无法删除已加载的dll

卸载程序集后可以正常删除

动态加载和卸载程序集用途

动态更新程序,功能封装到独立的dll库。

应用插件化支持,动态扩展功能。

相关推荐
吹牛不交税4 天前
.netcore项目部署在ubuntu22.04虚拟机的docker中的过程记录
docker·容器·.netcore
weixin_421994787 天前
基于 .NET 9.0 的高性能轻量级令牌桶限流服务
.net·.netcore·令牌桶
weixin_421994787 天前
MVC 模式初探
mvc·.net·.netcore
weixin_421994789 天前
互联网与 Web 应用简介
.net·.netcore
全栈小511 天前
【C#】合理使用DeepSeek相关AI应用为我们提供强有力的开发工具,在.net core 6.0框架下使用JsonNode动态解析json字符串,如何正确使用单问号和双问号做好空值处理
人工智能·c#·json·.netcore·deepseek
时光追逐者12 天前
C#/.NET/.NET Core优秀项目和框架2026年1月简报
c#·.net·.netcore
大尚来也14 天前
双库协同,各取所长:.NET Core 中 PostgreSQL 与 SQLite 的优雅融合实战
postgresql·sqlite·.netcore
吹牛不交税14 天前
admin.net-v2 框架使用笔记-netcore8.0/10.0版
vue.js·.netcore
张3蜂18 天前
java springboot2.0 api ;.netcore8 api ;python GunicornAPI ,哪种更强?请从多个维度,对比分析
java·python·.netcore
切糕师学AI19 天前
.NET Core Web 中的健康检查端点(Health Check Endpoint)
前端·kubernetes·.netcore