C#怎么动态加载DLL C# Assembly.LoadFile使用方法

8次阅读

Assembly.LoadFile 不推荐使用,因它仅加载单个 DLL 且不处理依赖、不参与默认加载上下文;应优先选用 Assembly.LoadFrom(兼容旧框架)或 AssemblyLoadContext(.NET Core+,支持卸载与隔离)。

C# 怎么动态加载 DLL C# Assembly.LoadFile 使用方法

Assembly.LoadFile 是 C# 中用于从指定文件路径加载程序集(DLL)的静态方法,但它 不推荐在常规场景中使用,尤其不适合动态插件、热更新或依赖解析复杂的场景。它只加载单个文件,不处理依赖、不参与默认加载上下文,容易引发 FileNotFoundException 或类型无法识别等问题。

LoadFile 的基本用法(仅作了解)

它直接按物理路径读取 DLL 文件并加载到当前进程,但 不会自动加载其引用的其他程序集,也不会触发 GAC 或 probing 机制:

  • 语法简单:Assembly assembly = Assembly.LoadFile(@"C:MyPluginMyLogic.dll");
  • 加载后可反射获取类型:Type t = assembly.GetType("MyLogic.Processor");
  • 不能用 typeof(T).Assembly == assembly 判断相等性(因 LoadFile 加载的程序集与普通引用程序集视为不同实例)

更安全的替代方案:AssemblyLoadContext(.NET Core/.NET 5+)

现代 .NET 推荐使用 AssemblyLoadContext 实现隔离、可卸载的动态加载:

  • 自定义上下文可避免污染默认上下文,支持显式卸载(context.Unload()
  • 示例:  var context = new AssemblyLoadContext(isCollectible: true);
      var assembly = context.LoadFromAssemblyPath(@”C:MyPluginMyLogic.dll”);
  • 依赖会尝试从同一上下文中解析,也可重写 Load(AssemblyName) 方法自定义查找逻辑

兼容旧框架的方案:Assembly.LoadFrom(.NET Framework / .NET Standard)

相比 LoadFileAssembly.LoadFrom 更可靠:

  • 它将 DLL 加入“加载上下文”,能自动探测并加载同目录下的依赖项
  • 支持多次调用同一路径(返回缓存实例),避免重复加载
  • 用法:Assembly assembly = Assembly.LoadFrom(@"C:MyPluginMyLogic.dll");
  • 注意:仍不能卸载,且路径必须是绝对路径或相对于 AppDomain.BaseDirectory 的相对路径

关键提醒:避免常见陷阱

无论选哪种方式,都要注意:

  • 确保目标 DLL 和所有依赖项版本匹配,特别是 System.* 或第三方库
  • 若 DLL 含有 NativeAOT 或 P/Invoke,需保证运行时环境一致(如 x64/x86)
  • 反射调用前建议检查 assembly.GetExportedTypes() 确认类型是否存在且可访问
  • 生产环境建议配合强命名(Strong Name)和签名验证提升安全性

基本上就这些。LoadFile 不复杂但容易忽略上下文问题,日常开发优先选 LoadFromAssemblyLoadContext,更稳也更可控。

text=ZqhQzanResources