0

0

.NET中的AssemblyLoadContext是什么?如何实现插件化架构?

畫卷琴夢

畫卷琴夢

发布时间:2025-11-16 10:56:03

|

911人浏览过

|

来源于php中文网

原创

AssemblyLoadContext是.NET中实现程序集隔离加载与卸载的核心机制,通过继承并重写Load方法可自定义上下文,利用isCollectible=true支持垃圾回收实现插件热插拔;需定义共享接口、动态加载插件DLL、反射实例化并确保无强引用以完成卸载,适用于构建模块化、可扩展的应用架构。

.net中的assemblyloadcontext是什么?如何实现插件化架构?

AssemblyLoadContext 是 .NET 中用于控制程序集(Assembly)加载和卸载的核心机制。与传统的 AppDomain 不同,.NET Core 和 .NET 5+ 移除了对多 AppDomain 的支持,取而代之的是 AssemblyLoadContext,它允许你以隔离的方式加载程序集,并在不需要时进行卸载(配合 GC 实现),非常适合实现插件化架构。

AssemblyLoadContext 的作用

默认情况下,.NET 程序使用一个默认的上下文来加载所有程序集,这些程序集一旦加载就无法单独卸载。而通过自定义 AssemblyLoadContext,你可以:

  • 隔离插件的程序集,避免版本冲突
  • 动态加载插件 DLL
  • 在插件不再需要时,卸载整个上下文及其加载的程序集
  • 控制依赖解析过程

如何创建自定义 AssemblyLoadContext

要实现插件隔离,通常需要继承 AssemblyLoadContext 并重写 Load 方法来处理依赖解析:

using System.Reflection;
using System.Runtime.Loader;

public class PluginLoadContext : AssemblyLoadContext
{
    private readonly AssemblyDependencyResolver _resolver;

    public PluginLoadContext(string pluginPath) : base(isCollectible: true)
    {
        _resolver = new AssemblyDependencyResolver(pluginPath);
    }

    protected override Assembly Load(AssemblyName assemblyName)
    {
        // 尝试从插件目录解析依赖
        string assemblyPath = _resolver.ResolveAssemblyToPath(assemblyName);
        if (assemblyPath != null)
        {
            return LoadFromAssemblyPath(assemblyPath);
        }
        return null;
    }
}

isCollectible: true 表示该上下文可以被垃圾回收,从而实现程序集卸载。

MedPeer
MedPeer

AI驱动的一站式科研服务平台

下载

实现插件化架构的步骤

要构建一个可插拔的应用程序,需遵循以下结构:

  1. 定义公共接口:主程序和插件之间通过共享的接口通信。通常放在一个独立的类库中(如 IPlugin.dll)。
  2. 插件实现接口:每个插件项目引用该接口,并实现具体逻辑。
  3. 主程序动态加载:扫描插件目录,使用自定义 AssemblyLoadContext 加载插件 DLL。
  4. 实例化并调用:通过反射创建插件实例并调用方法。
  5. 支持卸载:当不再需要插件时,释放引用,触发 GC 回收上下文。

示例代码(主程序加载插件):

var context = new PluginLoadContext(pluginDllPath);
Assembly assembly = context.LoadFromAssemblyPath(pluginDllPath);

Type pluginType = assembly.GetType("MyPlugin.Plugin");
IPlugin instance = (IPlugin)Activator.CreateInstance(pluginType);

instance.Execute();

// 卸载准备
context.Unload();
// 注意:需确保没有对该上下文中对象的强引用,否则无法回收

注意事项与最佳实践

  • 跨上下文传递对象时不能直接传实例,需使用 MarshalByRefObject 或序列化数据
  • 避免在插件中引用主程序的类型,应通过接口或消息解耦
  • 插件中的异常要妥善处理,防止崩溃主程序
  • 确保所有对插件对象的引用都被清除,才能成功卸载
  • 调试时可通过 GC.Collect()GC.WaitForPendingFinalizers() 触发卸载测试
基本上就这些。AssemblyLoadContext 提供了现代 .NET 中实现热插拔、模块化系统的基础能力,结合良好的接口设计,能构建出灵活稳定的插件架构。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

995

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

53

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

243

2025.12.29

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

138

2025.12.31

php网站源码教程大全
php网站源码教程大全

本专题整合了php网站源码相关教程,阅读专题下面的文章了解更多详细内容。

80

2025.12.31

视频文件格式
视频文件格式

本专题整合了视频文件格式相关内容,阅读专题下面的文章了解更多详细内容。

82

2025.12.31

不受国内限制的浏览器大全
不受国内限制的浏览器大全

想找真正自由、无限制的上网体验?本合集精选2025年最开放、隐私强、访问无阻的浏览器App,涵盖Tor、Brave、Via、X浏览器、Mullvad等高自由度工具。支持自定义搜索引擎、广告拦截、隐身模式及全球网站无障碍访问,部分更具备防追踪、去谷歌化、双内核切换等高级功能。无论日常浏览、隐私保护还是突破地域限制,总有一款适合你!

61

2025.12.31

出现404解决方法大全
出现404解决方法大全

本专题整合了404错误解决方法大全,阅读专题下面的文章了解更多详细内容。

458

2025.12.31

html5怎么播放视频
html5怎么播放视频

想让网页流畅播放视频?本合集详解HTML5视频播放核心方法!涵盖<video>标签基础用法、多格式兼容(MP4/WebM/OGV)、自定义播放控件、响应式适配及常见浏览器兼容问题解决方案。无需插件,纯前端实现高清视频嵌入,助你快速打造现代化网页视频体验。

16

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Java 教程
Java 教程

共578课时 | 41万人学习

国外Web开发全栈课程全集
国外Web开发全栈课程全集

共12课时 | 0.9万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号