ASP.NET MVC模块化开发——动态挂载外部项目
最近在开发一个MVC框架,开发过程中考虑到以后开发依托于框架的项目,为了框架的维护更新升级,代码肯定要和具体的业务工程分割开来,所以需要解决业务工程挂载在框架工程的问题,MVC与传统的ASP.NET不同,WebForm项目只需要挂在虚拟目录拷贝dll就可以访问,但是MVC不可能去引用工程项目的dll重新编译,从而产生了开发一个动态挂在MVC项目功能的想法,MVC项目挂载主要有几个问题,接下来进行详细的分析与完成解决方案
一般动态加载dll的方法是使用Assembly.LoadFIle的方法来调用,但是会存在如下问题:
1.如果MVC项目中存在依赖注入,框架层面无法将外部dll的类放入IOC容器
通过 BuildManager.AddReferencedAssembly方法在MVC项目启动前,动态将外部代码添加到项目的编译体系中,需要配合PreApplicationStartMethod注解使用,示例:
声明一个类,然后进行注解标记,指定MVC启动前方法
//使用PreApplicationStartMethod注解的作用是在mvc应用启动之前执行操作
[assembly: PreApplicationStartMethod(typeof(FastExecutor.Base.Util.PluginUtil), "PreInitialize")]
namespace FastExecutor.Base.Util
{
public class PluginUtil
{
public static void PreInitialize()
{ }
}
}
2.外部加载的dll中的Controller无法被识别
通过自定义的ControllerFactory重写GetControllerType方法进行识别
public class FastControllerFactory : DefaultControllerFactory
{ protected override Type GetControllerType(RequestContext requestContext, string controllerName)
{
Type ControllerType = PluginUtil.GetControllerType(controllerName + "Controller");
if (ControllerType == null)
{
ControllerType = base.GetControllerType(requestContext, controllerName);
}
return ControllerType;
}
}
在Global.asax文件中进行ControllerFactory的替换
ControllerBuilder.Current.SetControllerFactory(new FastControllerFactory());
ControllerTypeDic是遍历外部dll获取到的所有Controller,这里需要考虑到Controller通过RoutePrefix注解自定义Controller前缀的情况
IEnumerable<Assembly> assemblies = GetProjectAssemblies();
foreach (var assembly in assemblies)
{
foreach (var type in assembly.GetTypes())
{
if (type.GetInterface(typeof(IController).Name) != null && type.Name.Contains("Controller") && type.IsClass && !type.IsAbstract)
{
string Name = type.Name;
//如果有自定义的路由注解
if (type.IsDefined(typeof(System.Web.Mvc.RoutePrefixAttribute), false))
{
var rounteattribute = type.GetCustomAttributes(typeof(System.Web.Mvc.RoutePrefixAttribute), false).FirstOrDefault();
Name = ((System.Web.Mvc.RoutePrefixAttribute)rounteattribute).Prefix + "Controller";
}
if (!ControllerTypeDic.ContainsKey(Name))
{
ControllerTypeDic.Add(Name, type);
}
}
}
BuildManager.AddReferencedAssembly(assembly);
}
3.加载dll后如果要更新业务代码,dll会被锁定,无法替换,需要重启应用
解决办法是通过AppDomain对业务项目dll独立加载,更新时进行卸载
1)创建一个RemoteLoader一个可穿越边界的类,作为加载dll的一个包装
public class RemoteLoader : MarshalByRefObject
{
private Assembly assembly; public Assembly LoadAssembly(string fullName)
{
assembly = Assembly.LoadFile(fullName);
return assembly;
} public string FullName
{
get { return assembly.FullName; }
} }
2)创建LocalLoader作为AppDomian创建与卸载的载体
public class LocalLoader
{
private AppDomain appDomain;
private RemoteLoader remoteLoader;
private DirectoryInfo MainFolder;
public LocalLoader()
{ AppDomainSetup setup = AppDomain.CurrentDomain.SetupInformation; setup.ShadowCopyDirectories = setup.ApplicationBase;
appDomain = AppDomain.CreateDomain("PluginDomain", null, setup); string name = Assembly.GetExecutingAssembly().GetName().FullName;
remoteLoader = (RemoteLoader)appDomain.CreateInstanceAndUnwrap(
name,
typeof(RemoteLoader).FullName);
} public Assembly LoadAssembly(string fullName)
{
return remoteLoader.LoadAssembly(fullName);
} public void Unload()
{
AppDomain.Unload(appDomain);
appDomain = null;
} public string FullName
{
get
{
return remoteLoader.FullName;
}
}
}
这里需要说明的,AppDomainSetup配置文件请使用AppDomain.CurrentDomain.SetupInformation也就是使用框架的作用于配置信息,因为业务代码会引用到很多框架的dll,如果独立创建配置信息,会有找不到相关dll的错误,同时这里也需要配置web.confg文件指定额外的dll搜索目录,因为业务工程代码也会有很多层多个dll相互引用,不指定目录也会存在找不到依赖dll的错误
<runtime>
<assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1">
<!--插件加载目录-->
<probing privatePath="PluginTemp" />
</assemblyBinding>
</runtime>
3)创建业务代码文件夹Plugin与临时dll文件夹PluginTemp
为什么要创建临时文件夹呢,因为我们需要在PluginTemp真正的加载dll,然后监听Plugin文件夹的文件变化,有变化时进行AppDomain卸载这个操作,将Plugin中的dll拷贝到PluginTemp文件夹中,再重新加载dll
监听Plugin文件夹:
private static readonly FileSystemWatcher _FileSystemWatcher = new FileSystemWatcher();
public static void StartWatch()
{
_FileSystemWatcher.Path = HostingEnvironment.MapPath("~/Plugin");
_FileSystemWatcher.Filter = "*.dll";
_FileSystemWatcher.Changed += _fileSystemWatcher_Changed; _FileSystemWatcher.IncludeSubdirectories = true;
_FileSystemWatcher.NotifyFilter =
NotifyFilters.Attributes | NotifyFilters.CreationTime | NotifyFilters.DirectoryName | NotifyFilters.FileName | NotifyFilters.LastAccess | NotifyFilters.LastWrite | NotifyFilters.Security | NotifyFilters.Size;
_FileSystemWatcher.EnableRaisingEvents = true;
}
private static void _fileSystemWatcher_Changed(object sender, FileSystemEventArgs e)
{
DllList.Clear();
Initialize(false);
InjectUtil.InjectProject();
}
拷贝dll:
if (PluginLoader == null)
{
PluginLoader = new LocalLoader();
}
else
{
PluginLoader.Unload();
PluginLoader = new LocalLoader();
} TempPluginFolder.Attributes = FileAttributes.Normal & FileAttributes.Directory;
PluginFolder.Attributes = FileAttributes.Normal & FileAttributes.Directory;
//清理临时文件。
foreach (var file in TempPluginFolder.GetFiles("*.dll", SearchOption.AllDirectories))
{
try
{
File.SetAttributes(file.FullName, FileAttributes.Normal);
file.Delete();
}
catch (Exception)
{
//这里虽然能成功删除,但是会报没有权限的异常,就不catch了
} }
//复制插件进临时文件夹。
foreach (var plugin in PluginFolder.GetFiles("*.dll", SearchOption.AllDirectories))
{
try
{
string CopyFilePath = Path.Combine(TempPluginFolder.FullName, plugin.Name);
File.Copy(plugin.FullName, CopyFilePath, true);
File.SetAttributes(CopyFilePath, FileAttributes.Normal);
}
catch (Exception)
{
//这里虽然能成功删除,但是会报没有权限的异常,就不catch了
}
}
注:这里有个问题一直没解决,就是删除文件拷贝文件的时候,AppDomain已经卸载,但是始终提示无权限错误,但是操作又是成功的,暂时还未解决,如果大家有解决方法可以一起交流下
加载dll:
public static IEnumerable<Assembly> GetProjectAssemblies()
{
if (DllList.Count==)
{
IEnumerable<Assembly> assemblies = TempPluginFolder.GetFiles("*.dll", SearchOption.AllDirectories).Select(x => PluginLoader.LoadAssembly(x.FullName));
foreach (Assembly item in assemblies)
{
DllList.Add(item);
}
}
return DllList;
}
4.业务代码的cshtml页面如何加入到框架中被访问
在MVC工程中,cshtml也是需要被编译的,我们可以通过RazorBuildProvider将外部编译的页面动态加载进去
public static void InitializeView()
{
IEnumerable<Assembly> assemblies = GetProjectAssemblies();
foreach (var assembly in assemblies)
{
RazorBuildProvider.CodeGenerationStarted += (object sender, EventArgs e) =>
{
RazorBuildProvider provider = (RazorBuildProvider)sender;
provider.AssemblyBuilder.AddAssemblyReference(assembly);
};
} }
RazorBuildProvider方法啊只是在路由层面将cshtml加入到框架中,我们还需要将业务工程View中模块的页面挂载虚拟目录到框架中,如图所示
5.框架启动后,更新业务dll带来的相关问题
在启动的项目中我们更新dll,我们希望达到的效果是和更新框架bin目录文件的dll一样,程序会重启,这样就会再次调用被PreApplicationStartMethod注解标注的方法,不需要在代码中做额外处理判断是首次加载还是更新加载,同时也做不到动态的将外部dll加入到MVC编译dll体系中,也只能启动前加载,查了很多资料,重新加载项目可以通过代码控制IIS回收程序池达到效果,但是因为各种繁琐的权限配置问题而放弃,我最后的解决方法是比较歪门邪道的方法,更新web.config文件的修改日期,因为iis会监控配置文件,更新了会重启引用,大家如果有更好的简单的方法,可以评论回复我呦
//这里通过修改webconfig文件的时间达到重启应用,加载项目dll的目的!
File.SetLastWriteTime(HostingEnvironment.MapPath("~/Web.config"), System.DateTime.Now);
博客园没找到资源上传地址,传到码云上了,放个地址:https://gitee.com/grassprogramming/FastExecutor/attach_files
ASP.NET MVC模块化开发——动态挂载外部项目的更多相关文章
- Asp.net Mvc模块化开发之分区扩展框架
对于一个企业级项目开发,模块化是非常重要的. 默认Mvc框架的AreaRegistration对模块化开发真的支持很好吗?真的有很多复杂系统在使用默认的分区开发的吗?我相信大部分asp.net的技术团 ...
- Asp.net Mvc模块化开发系列(目录)
模块化开发是非常重要的,模块化开发是个系统性问题,为此我觉得有必须要写一个系列的文章才能基本说的清楚 那又为什么要写一个目录呢? 其一.是对我昨天承诺写一个系列新的文章的回应 其二.是先写出一个大纲, ...
- asp.net MVC提高开发速度(创建项目模板)
- Asp.net Mvc模块化开发之“开启模块开发、调试的简单愉快之旅”
整个世界林林种种,把所有的事情都划分为对立的两个面. 每个人都渴望的财富划分为富有和贫穷,身高被划分为高和矮,身材被划分为胖和瘦,等等. 我们总是感叹,有钱人的生活我不懂;有钱人又何尝能懂我们每天起早 ...
- Asp.net Mvc模块化开发之“部分版本部分模块更新(上线)”
项目开发从来就不是一个简单的问题.更难的问题是维护其他人开发的项目,并且要修改bug.如果原系统有重大问题还需要重构. 怎么重构系统不是本文探讨的问题,但是重构后如何上线部署和本文关系密切.这个大家可 ...
- net Mvc模块化开发
Asp.net Mvc模块化开发之“部分版本部分模块更新(上线)” 项目开发从来就不是一个简单的问题.更难的问题是维护其他人开发的项目,并且要修改bug.如果原系统有重大问题还需要重构. 怎么重构系统 ...
- 全面解析ASP.NET MVC模块化架构方案
什么叫架构?揭开架构神秘的面纱,无非就是:分层+模块化.任意复杂的架构,你也会发现架构师也就做了这两件事. 本文将会全面的介绍我们团队在模块化设计方面取得的经验.之所以加了“全面”二字,是因为本文的内 ...
- ASP.NETCORE MVC模块化
ASP.NETCORE MVC模块化编程 前言 记得上一篇博客中跟大家分享的是基于ASP.NETMVC5,实际也就是基于NETFRAMEWORK平台实现的这么一个轻量级插件式框架.那么今天我主要分享的 ...
- Easyui + asp.net mvc + sqlite 开发教程(录屏)适合入门
Easyui + asp.net mvc + sqlite 开发教程(录屏)适合入门 第一节: 前言(技术简介) EasyUI 是一套 js的前端框架 利用它可以快速的开发出好看的 前端系统 web ...
随机推荐
- centos7.5优化系统脚本(虚拟机下)
#!/usr/bin/bash #安装常用软件,首先必须自行调整好网卡配置文件,保证可以上网,否则,下列优化会失败 yum -y install wget vim lrzsz bash-complet ...
- Python 实现转堆排序算法原理及时间复杂度(多图解释)
原创文章出自公众号:「码农富哥」,欢迎转载和关注,如转载请注明出处! 堆基本概念 堆排序是一个很重要的排序算法,它是高效率的排序算法,复杂度是O(nlogn),堆排序不仅是面试进场考的重点,而且在很多 ...
- 单页应用(SPA,Single-page-App)和多页应用(MPA,Multi-page App)的区别
单页应用(SPA,Single-page-App)和多页应用(MPA,Multi-page App)的区别 参考博客:https://www.jianshu.com/p/4c9c29967dd6
- 优秀 .NET 开源项目集锦
Github 地址: https://github.com/jasonhua95/awesome-dotnet-core awesome-dotnet-core .NET Core框架.库和软件的中文 ...
- css如何玩转有序无序列表项list样式
在无序列表ul>li中,无线列表的标志是出现在各列表前面的圆点.在有序列表ol>li中,前面默认带有数字,如何修改列表前面的项目符号,只需要通过list-style调整就好,常见的符号有( ...
- 5.Android-电话拨号器详解
之前学习了3.Android-ADT之helloworld项目结构介绍后,本章便来写个简单的电话拨号器程序. 实现的步骤如下所示: 1.创建项目 2.写layout/activity_main.xml ...
- webStorm 2019 激活码,phpStorm 2019激活,idea激活,pyCharm激活【永久使用】
[2020-01-16 亲测可用] 无废话版!----直接激活 [麻烦激活后,在评论发表:eg:2020-01-11 09:00 测试可用],有问题直接反馈 我及时修改,建议收藏此博客 都能永久激活, ...
- 错误:EfficientDet网络出现"No boxes to NMS"并且mAP:0.0的解决方案
近日,在使用谷歌新推出来的一个网络EfficientDet进行目标检测训练自己的数据集的时候,出现了如下错误: 其中项目开源地址是:https://github.com/toandaominh1997 ...
- LAMP环境搭建+配置虚拟域名
Centos下PHP,Apache,Mysql 的安装 安装Apache yum -y install httpd systemctl start httpd 添加防火墙 firewall-cmd - ...
- MAC使用命令行解压rar
使用homebrew安装unrar brew install unrar 安装完成后cd到rar文件目录,使用终端命令解压 unrar x 需要解压的文件