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

news/2024/5/19 22:25:51 标签: .netcore

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

AssemblyLoadContext

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

创建一个 .Net 库项目

    public class Class1
    {
        public void Fn1()
        {
            Console.WriteLine("Fn1");
        }

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

创建一个控制台项目

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

    internal class MyAssemblyContext:AssemblyLoadContext
    {
        public MyAssemblyContext() : base(isCollectible: true)
        { }
    }

测试加载和卸载方法

        //告诉编译器不要内联该方法,防止栈上存在引用导致无法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");
        }

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

            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库。

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


http://www.niftyadmin.cn/n/4942683.html

相关文章

YOLOv5算法改进(1)— 如何去改进YOLOv5算法

前言&#xff1a;Hello大家好&#xff0c;我是小哥谈。YOLOv5基础知识入门系列、YOLOv5源码中的参数超详细解析系列和YOLOv5入门实践系列学习完成之后&#xff0c;接着就进入YOLOv5进阶改进算法系列了。&#x1f389;为了让大家能够清楚地了解如何去改进YOLOv5算法以及从哪几方…

FPGA:uart原理+tx发送模块

文章目录 一、串口通信二、UART通信三、tx发送模块 一、串口通信 处理器与外部设备通信的两种方式&#xff1a; 串行通信&#xff1a; 指数据的各个位使用多条数据线同时进行传输。 并行通信&#xff1a; 将数据分成一位一位的形式在一条数据线上逐个传输。 串行通信的通信方…

Java常用API---快速达到Java工作水准系列(1)

目录 1.集合 2.包装类 3.日期处理以及格式化 4.字符串处理类 5.数组 5.BigDecimal 6.Math 1.集合 毋庸置疑&#xff0c;集合在实际项目的使用概率几乎是百分之百。无论是用于数据存储和管理、去重和查找亦或是数据检索和遍历&#xff0c;都离不开集合的使用。任何一个项…

【枚举gcd】CF803 C

Problem - C - Codeforces 题意&#xff1a; 思路&#xff1a; 要使所有数的 gcd 尽可能大 设 gcd k 那么就是 k1 * k&#xff0c;k2 * k&#xff0c;k3 * k&#xff0c;.... 它们的和是n 那么(k1 k2 k3 ....) * k n 根据惯用套路&#xff0c;我们去枚举 n 的因子即…

【Spring】统一事件的处理(拦截器、统一异常处理、统一数据格式返回)

文章目录 前言一、Spring 拦截器1.1 用户登录权限校验案例1.1.1 最初的用户登录验证1.1.2 使用 Spring AOP 实现登录验证的问题 1.2 Spring 拦截器的使用1.2.1 Spring 拦截器概念与使用步骤1.2.2 使用拦截器实现对用户登录权限的校验 1.3 拦截器实现原理1.4 Spring 拦截器和 Sp…

热电联产在综合能源系统中的选址定容研究(matlab代码)

目录 1 主要内容 目标函数 程序模型 2 部分代码 3 程序结果 1 主要内容 该程序参考《热电联产在区域综合能源系统中的定容选址研究》&#xff0c;主要针对电热综合能源系统进行优化&#xff0c;确定热电联产机组的位置和容量&#xff0c;程序以33节点电网和17节点热网为例…

【Linux】【驱动】杂项设备驱动

【Linux】【驱动】杂项设备驱动 Linux三大设备驱动1. 我们这节课要讲的杂项设备驱动是属于我们这三大设备驱动里面的哪个呢?2.杂项设备除了比字符设备代码简单&#xff0c;还有别的区别吗?3.主设备号和次设备号是什么? 挂载驱动 杂项设备驱动是字符设备驱动的一种&#xff0…

Java课题笔记~ SpringMVC注解式开发

2.1 Controller 传统的配置式开发中的控制器Controller类必须实现Controller接口&#xff0c;并实现接口中的HandleRequest()方法&#xff0c;还需要再配置文件中配置处理器映射&#xff0c;且一个处理器&#xff08;控制器&#xff09;只能有一个方法&#xff0c;为了实现程序…