aspnetcore插件开发dll热加载
该项目比较简单,只是单纯的把业务的dll模块和controller的dll做了一个动态的添加删除处理,目的就是插件开发。由于该项目过于简单,请勿吐槽。复杂的后续可以通过泛型的实体、dto等做业务和接口的动态区分。
项目结构如下:

上面的两个模块是独立通过dll加载道项目中的

repository动态的核心思想在此项目中是反射
public interface IRepositoryProvider
{
IRepository GetRepository(string serviceeName);
}
public class RepositoryProvider : IRepositoryProvider
{
public IRepository GetRepository(string x)
{
var path = $"{Directory.GetCurrentDirectory()}\\lib\\{x}.Repository.dll";
var _AssemblyLoadContext = new AssemblyLoadContext(Guid.NewGuid().ToString("N"), true);
Assembly assembly = null;
using (var fs = new FileStream(path, FileMode.Open, FileAccess.Read))
{
assembly = _AssemblyLoadContext.LoadFromStream(fs);
} //var assembly = Assembly.LoadFrom(path);
var types = assembly.GetTypes()
.Where(t => typeof(IRepository).IsAssignableFrom(t) && !t.IsInterface);
return (IRepository)Activator.CreateInstance(types.First());
}
}
通过一个provider注入来获取示例,这个repository的示例既然是动态热拔插,能想到暂时只能是反射来做这一块了。
using Autofac;
using IOrder.Repository;
using Order.Repository; namespace AutofacRegister
{
public class RepositoryModule:Module
{
protected override void Load(ContainerBuilder builder)
{
//builder.RegisterType<Repository>().As<IRepository>().SingleInstance();
builder.RegisterType<RepositoryProvider>().As<IRepositoryProvider>().InstancePerLifetimeScope();
}
}
}
controller插件这一块大同小异,这个控制器是通过程序集注入来实现的
public class MyControllerFilter : IStartupFilter
{ private readonly PluginManager pluginManager;
List<string> controllers = new List<string> { "First","Second" };
public MyControllerFilter(PluginManager pluginManager)
{
this.pluginManager = pluginManager;
controllers.ForEach(x => pluginManager.LoadPlugins($"{Directory.GetCurrentDirectory()}\\lib\\", $"{x}.Impl.dll"));
}
Action<IApplicationBuilder> IStartupFilter.Configure(Action<IApplicationBuilder> next)
{
BindingFlags bindingFlags = BindingFlags.Instance | BindingFlags.Public | BindingFlags.DeclaredOnly;
return app =>
{ app.UseRouting();
app.UseEndpoints(endpoints =>
{
foreach (IPlugin item in pluginManager.GetPlugins())
{
foreach (MethodInfo mi in item.GetType().GetMethods(bindingFlags))
{
endpoints.MapPost($"/{item.GetType().Name.Replace("Service", "")}/{mi.Name}", async (string parameters, HttpContext cotext) =>
{ var task = (Task)mi.Invoke(item, new object[] { parameters });
if (task is Task apiTask)
{
await apiTask; // 如果任务有返回结果
if (apiTask is Task<object> resultTask)
{
var res = await resultTask;
return Results.Ok(JsonConvert.SerializeObject(res));
}
} // 如果方法没有返回 Task<ApiResult>,返回 NotFound
return Results.NotFound("Method execution did not return a result.");
});
}
}
});
next(app);
};
}
}
但是有一个问题,它的变化势必需要重新渲染整个controller,我只能重启他的服务了。
using Microsoft.Extensions.Hosting;
using System;
using System.Diagnostics;
using System.IO;
using System.Threading;
using System.Threading.Tasks; namespace StartupDiagnostics
{
public class FileWatcherService : IHostedService
{
private readonly string _watchedFolder;
private FileSystemWatcher _fileSystemWatcher;
private readonly IHostApplicationLifetime _appLifetime;
public FileWatcherService(IHostApplicationLifetime appLifetime)
{
_watchedFolder =Path.Combine(Directory.GetCurrentDirectory(),"lib"); //细化指定类型的dll
_appLifetime = appLifetime;
} public Task StartAsync(CancellationToken cancellationToken)
{
_fileSystemWatcher = new FileSystemWatcher(_watchedFolder);
_fileSystemWatcher.NotifyFilter = NotifyFilters.LastWrite | NotifyFilters.FileName | NotifyFilters.DirectoryName;
_fileSystemWatcher.Changed += OnFileChanged;
_fileSystemWatcher.Created += OnFileChanged;
_fileSystemWatcher.Deleted += OnFileChanged;
_fileSystemWatcher.Renamed += OnFileChanged;
_fileSystemWatcher.EnableRaisingEvents = true; return Task.CompletedTask;
} public Task StopAsync(CancellationToken cancellationToken)
{
_fileSystemWatcher.Dispose();
return Task.CompletedTask;
} private void OnFileChanged(object sender, FileSystemEventArgs e)
{
// 文件夹内容发生变化时重新启动应用程序
var processStartInfo = new ProcessStartInfo
{
FileName = "dotnet",
Arguments = $"exec \"{System.Reflection.Assembly.GetEntryAssembly().Location}\"",
UseShellExecute = false
}; _appLifetime.ApplicationStopped.Register(() =>
{
Process.Start(processStartInfo);
});
_appLifetime.StopApplication(); }
}
}
repository这一块页面效果没法展示,
controllerr可以通过swagger来看看,first和second这可以通过删除dll和添加dll来增加和删除controller给第三方。


这是控制台展示的重启效果

源代码如下:
liuzhixin405/AspNetCoreSimpleAop (github.com)
aspnetcore插件开发dll热加载的更多相关文章
- Aspnetcore下面服务器热更新与配置热加载
原文:Aspnetcore下面服务器热更新与配置热加载 Asp.net的热更新方案Appdomain在aspnetcore中不被支持了 新的方案如下: 配置文件更新选项 reloadOnChange ...
- Windows7 安装vs2015 之后 调试Web项目IIS启动不了 aspnetcore.dll未能加载
安装windows企业版,整整折腾了两天了,一个本身家里网络环境不好,时不时掉线,终于披荆斩棘,克服了所有困难,结果VS2015 EnterPrise 版本在调试Web环境的时候,始终在任务栏里找不到 ...
- 模块 DLL C:\WINDOWS\system32\inetsrv\aspnetcore.dll 未能加载。返回的数据为错误信息。
更新了win10的版本后,就启动原来的iis发布的程序 程序池就自动关闭.后来 启动网站 iis程序池自动关闭. 在为应用程序池“.NET v4.5”提供服务的工作进程“21908”中,协议“http ...
- 使用 .NET Core 3.0 的 AssemblyLoadContext 实现插件热加载
一般情况下,一个 .NET 程序集加载到程序中以后,它的类型信息以及原生代码等数据会一直保留在内存中,.NET 运行时无法回收它们,如果我们要实现插件热加载 (例如 Razor 或 Aspx 模版的热 ...
- IE报错:模块"scrrun.dll"已加载,但对DllRegisterServer的调用失败,错误代码为0x80004005
在我的win10系统上打开某内部网页登录的时候弹出'模块"scrrun.dll"已加载,但对DllRegisterServer的调用失败,错误代码为0x80004005'报错信息, ...
- webpack 教程 那些事儿03-webpack两大精华插件,热加载
本节主要讲述 webpack的两大经典开发调试插件,热插拔内存缓存机制 文章目录 1. html-webpack-plugin插件的使用 2. webpack-dev-middleware 插件登场 ...
- [Eclipse] - 集成JBoss7热加载和自动发布
使用Eclipse + JBoss开发时,总是要重启项目或JBoss,烦人.下面方法可以很简单的实现Eclipse + JBoss热加载和自动发布. 我的环境是JBoss 7.1.1 Final 1) ...
- [Eclipse] - 集成Tomcat热加载插件
使用Eclipse + Tomcat,要使用热加载,总是会重启tomcat webapp. 可以使用这个插件:jrebel 如果是Tomcat 7.0+版本,需要使用jrebel5.5.1+的版本,不 ...
- Eclipse tomcat插件禁用热加载
Eclipse中的tomcat插件默认是开户了热加载,只要是修改了java文件一保存,tomcat自动编译.加载.发布,很吃内存. 关闭方法: 打开eclipse,找到server项: 双击打开,修改 ...
- mybatis热加载的实现
最近在使用mybatis,由于是刚刚开始用,用的并不顺手,目前是感觉有2个地方非常的不好用: 1.mybatis调试不方便 由于dao层只有接口,实现只是一个map的xml文件,想加断点都没有地方加, ...
随机推荐
- CDC报错:invalid value null used for required field "null"
原因:数据库ceate_time时间字段设置为not null ,但是没有设置default值解决办法:1.修改ceate_time字段可以为null,即去掉 `create_time` dateti ...
- KingbaseES 判断字符串包含的几种方法
判断字符串包含的几种方法: 1. position test=# select position('aa' in 'abcd'); position ---------- 0 (1 行记录) test ...
- 性能测试思想(What is performance testing?)
1.什么是性能测试 什么是软件性能? 定义:软件的性能是软件的一种非功能特性,它关注的不是软件是否能够完成特定的功能,而是在完成该功能是展示出来的及时性. 比如:一个登录功能他能实现登录操作,但是登录 ...
- Android相对布局(来自菜鸟教程)
- 使用OHOS SDK构建vorbis
参照OHOS IDE和SDK的安装方法配置好开发环境. 从github下载源码. 执行如下命令: git clone --depth=1 https://github.com/xiph/vorbis ...
- 【直播回顾】OpenHarmony知识赋能五期第五课——多媒体子系统之视频解读
5月19日晚上19点,知识赋能第五期第五节课<OpenHarmony标准系统多媒体子系统之视频解读>,在OpenHarmony开发者成长计划社群内成功举行. 本期课程,由深开鸿资深技术专家 ...
- Makefile 基础教程:从零开始学习
在软件开发过程中,Makefile是一个非常重要的工具,它可以帮助我们自动构建程序,管理程序依赖关系,提高开发效率.本篇博客将从基础开始,介绍Makefile的相关知识,帮助大家快速掌握Makefil ...
- Linux 使用 crontab 定时拆分日志、清理过期文件
@ 目录 前言 简介 一.准备工作 1.1 创建写入脚本 1.2 设置执行权限 1.3 添加定时任务 1.4 配置生效 二.Tomcat日志 按每天分割 2.1 创建一个 sh文件 2.2 设置执行权 ...
- 如何跑各种check
如何进行 Fastcheck? 首先,导入环境变量: export CODE_BASE=/data/openGauss-server export BINARYLIBS=/data/openGauss ...
- Sample上新,从API 8开始支持!速来拿走
原文:https://mp.weixin.qq.com/s/TxUOSXySZRwQaECenxt-Og ,点击链接查看更多技术内容. 搭载API 8的新SDK已经发布.围绕着新SDK,官方贴心地输出 ...