0

0

c# 如何动态加载程序集

煙雲

煙雲

发布时间:2026-01-02 11:42:38

|

478人浏览过

|

来源于php中文网

原创

.NET Core/.NET 5+ 中应使用可卸载的 AssemblyLoadContext(isCollectible: true)替代不可卸载的 Assembly.LoadFrom;需重写 Load 方法、显式调用 Unload、配合 AssemblyDependencyResolver 解析依赖,并注意类型可见性与动态程序集限制。

c# 如何动态加载程序集

AssemblyLoadContext.Load 适用于 .NET Core / .NET 5+ 的隔离加载

在 .NET Core 及更高版本中,Assembly.LoadFrom 会把程序集加载进默认上下文,无法卸载,容易造成内存泄漏。真正可卸载的动态加载必须用 AssemblyLoadContext

关键点:

  • 继承 AssemblyLoadContext 并重写 Load 方法,控制依赖解析逻辑
  • 新建实例时传 isCollectible: true,否则仍不可卸载
  • 加载后必须显式调用 context.Unload(),且需确保无任何托管引用残留(包括事件订阅、静态字段、委托缓存)
var context = new AssemblyLoadContext(isCollectible: true);
try
{
    var asm = context.LoadFromAssemblyPath(@"C:\plugins\MyPlugin.dll");
    // 使用 asm 创建类型实例等...
}
finally
{
    context.Unload(); // 必须调用,但可能阻塞直到 GC 回收完成
}

Assembly.LoadFrom 在 .NET Framework 中最常用但不可卸载

如果你还在用 .NET Framework(如 4.8),Assembly.LoadFrom 是最直接的方式,但它会把程序集加载进 Default 上下文,整个进程生命周期内无法释放。

常见误用:

  • 反复调用 LoadFrom 同一路径 → 抛出 FileLoadException:“不能为同一个程序集加载多个版本”
  • 试图用 AppDomain.Unload 卸载 → .NET Framework 中仅对非默认 AppDomain 有效,且已标记为过时
try
{
    var asm = Assembly.LoadFrom(@"C:\legacy\Tool.dll");
    var type = asm.GetType("Tool.Processor");
    var inst = Activator.CreateInstance(type);
}
catch (FileLoadException ex) when (ex.Message.Contains("same assembly"))
{
    // 已加载,从 AppDomain.CurrentDomain.GetAssemblies() 查找复用
}

使用 AssemblyDependencyResolver 避免 MissingMethodException

动态加载的程序集若依赖其他 DLL(比如 Newtonsoft.Json 或自定义基础库),不处理依赖会导致运行时报 MissingMethodExceptionFileNotFoundException

沙之丘企业网站程序3.5
沙之丘企业网站程序3.5

沙之丘企业网站程序是一个以asp.net(C#) 4.0 +access进行开发的企业网站源码。主要功能:1、产品、设备、新闻系统2、留言信息直接发邮件到相关部门3、所有链接都以一级目录显示更好的权重4、其他信息扩展,可以增加如:人事招聘,公司介绍,地图,联系我们等5、带有商品和设备的搜索功能6、模板动态化方便扩展模板7、简体繁体选择显示运行环境:windows 2003或者更高windows服务

下载

AssemblyDependencyResolver 能自动读取 .deps.json 文件(发布时生成),定位依赖路径:

  • 只在 .NET Core 3.0+ 可用,需确保目标程序集是通过 dotnet publish 输出的
  • 构造时传入主程序集路径(不是被加载 DLL 的路径),resolver 才能正确解析依赖树
  • 配合自定义 AssemblyLoadContextLoad 方法使用,否则依赖仍会 fallback 到默认上下文
var resolver = new AssemblyDependencyResolver(assemblyPath); // 注意:传的是 MyPlugin.deps.json 同级的主程序集路径
var context = new CustomLoadContext(resolver); // 自定义 Load 方法中调用 resolver.ResolveAssemblyToPath(...)

反射调用前务必检查 Assembly.IsDynamic 和目标类型可见性

动态加载后,即使 Assembly 对象存在,也可能因以下原因导致 GetType 返回 null 或 Activator.CreateInstance 失败:

  • asm.IsDynamic 为 true → 说明是 AssemblyBuilder 生成的动态程序集,不能用 LoadFrom 加载方式获取
  • 目标类型是 internal 且未用 [InternalsVisibleTo] 开放给调用方程序集
  • 类型位于嵌套命名空间或泛型定义中,GetType("Name") 写法不完整(应为 "NS.Outer+Inner`1[[T]]"

建议先遍历 asm.GetTypes() 确认类型是否存在,再用 BindingFlags.NonPublic | BindingFlags.Public 获取构造器。

加载本身不难,难的是依赖路径、上下文生命周期和类型可见性的组合判断——这三个地方错一个,错误现象就可能完全不同。

相关专题

更多
json数据格式
json数据格式

JSON是一种轻量级的数据交换格式。本专题为大家带来json数据格式相关文章,帮助大家解决问题。

403

2023.08.07

json是什么
json是什么

JSON是一种轻量级的数据交换格式,具有简洁、易读、跨平台和语言的特点,JSON数据是通过键值对的方式进行组织,其中键是字符串,值可以是字符串、数值、布尔值、数组、对象或者null,在Web开发、数据交换和配置文件等方面得到广泛应用。本专题为大家提供json相关的文章、下载、课程内容,供大家免费下载体验。

528

2023.08.23

jquery怎么操作json
jquery怎么操作json

操作的方法有:1、“$.parseJSON(jsonString)”2、“$.getJSON(url, data, success)”;3、“$.each(obj, callback)”;4、“$.ajax()”。更多jquery怎么操作json的详细内容,可以访问本专题下面的文章。

307

2023.10.13

go语言处理json数据方法
go语言处理json数据方法

本专题整合了go语言中处理json数据方法,阅读专题下面的文章了解更多详细内容。

74

2025.09.10

c语言中null和NULL的区别
c语言中null和NULL的区别

c语言中null和NULL的区别是:null是C语言中的一个宏定义,通常用来表示一个空指针,可以用于初始化指针变量,或者在条件语句中判断指针是否为空;NULL是C语言中的一个预定义常量,通常用来表示一个空值,用于表示一个空的指针、空的指针数组或者空的结构体指针。

229

2023.09.22

java中null的用法
java中null的用法

在Java中,null表示一个引用类型的变量不指向任何对象。可以将null赋值给任何引用类型的变量,包括类、接口、数组、字符串等。想了解更多null的相关内容,可以阅读本专题下面的文章。

434

2024.03.01

default gateway怎么配置
default gateway怎么配置

配置default gateway的步骤:1、了解网络环境;2、获取路由器IP地址;3、登录路由器管理界面;4、找到并配置WAN口设置;5、配置默认网关;6、保存设置并退出;7、检查网络连接是否正常。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

219

2023.12.07

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

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

65

2025.12.31

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

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

44

2025.12.31

热门下载

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

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
WEB前端教程【HTML5+CSS3+JS】
WEB前端教程【HTML5+CSS3+JS】

共101课时 | 8.1万人学习

JS进阶与BootStrap学习
JS进阶与BootStrap学习

共39课时 | 3.1万人学习

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

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