使用ASP.NET MVC引擎開發(fā)插件系統(tǒng)
一、前言
我心中的插件系統(tǒng)應該是像Nop那樣(更牛的如Orchard,OSGI.NET),每個插件模塊不只是一堆實現(xiàn)了某個業(yè)務接口的dll,然后采用反射或IOC技術來調用,而是一個完整的mvc小應用,我可以在后臺控制插件的安裝和禁用,目錄結構如下:

生成后放在站點根目錄下的Plugins文件夾中,每個插件有一個子文件夾
Plugins/Sms.AliYun/
Plugins/Sms.ManDao/
我是一個有強迫癥的的懶人,我不想將生成的dll文件拷貝到bin目錄。
二、要解決的問題
1.asp.net引擎默認只會加載“bin”文件夾中的dll,而我們想要的插件文件則是分散在Plugins目錄下的各個子目錄中。
2.視圖中使用了模型時如何處理?默認情況下RazorViewEngine使用BuildManager將視圖編譯成動態(tài)程序集,然后使用Activator.CreateInstance實例化新編譯的對象,而使用插件dll時,當前的AppDomain不知道如何解析這種引用了模型的視圖,因為它不存在于“bin”或GAC中。更糟糕的是,不會收到任何錯誤消息,告訴您為什么它不工作,或者問題在哪。相反,他會告訴你,從View目錄中找不到文件。
3.某個插件正掛在站點下運行著,直接覆蓋插件的dll,會告訴你當前dll正在使用,不能被覆蓋。
4.視圖文件不放站點的View目錄中,該如何加載。
三.Net 4.0讓這一切變成可能
Net4.0有一個新特性是在應用程序初始化之前執(zhí)行代碼的能力(PreApplicationStartMethodAttribute),這個特性使得應用程序在Application_Star前可以做一些工作,例如我們可以在應用啟動之前告知我們的mvc插件系統(tǒng)的dll放在哪,做預加載處理等。關于.net的幾個新特性,有歪果仁寫得有博客來介紹,點我。,關于PreApplicationStartMethodAttribute,有博友已經(jīng)寫過了,點我。 Abp的啟動模塊應該也是使用PreApplicationStartMethodAttribute這個特性原理來實現(xiàn)的,具體是不是這樣還沒看。
四、解決方案
1.修改主站點web.config目錄,讓運行時除了加載bin目錄中的文件,還可以從其它目錄加載
<runtime> <assemblyBinding xmlns="urn:schemas-microsoft-com:asm.v1"> <probing privatePath="Plugins/temp/" /> </assemblyBinding> </runtime>
2.開發(fā)一個簡易的插件管理類,這個類的作用就是在Application_Start之前就把Plugins各子目錄中的dll拷貝到第1步指定的文件夾中,為了讓demo盡可能簡單,沒有對重復的dll進行檢測(比如插件中引用了ef程序集,主站點也引用了,在站點bin目錄中已經(jīng)存在ef的dll了,就沒必要再把插件中的dll拷貝到上面設置的動態(tài)程序集目錄中)
using System;
using System.Collections.Generic;
using System.IO;
using System.Linq;
using System.Reflection;
using System.Text;
using System.Threading.Tasks;
using System.Web;
using System.Web.Compilation;
using System.Web.Hosting;
[assembly: PreApplicationStartMethod(typeof(Plugins.Core.PreApplicationInit), "Initialize")]
namespace Plugins.Core
{
public class PreApplicationInit
{
static PreApplicationInit()
{
PluginFolder = new DirectoryInfo(HostingEnvironment.MapPath("~/plugins"));
ShadowCopyFolder = new DirectoryInfo(HostingEnvironment.MapPath("~/plugins/temp"));
}
/// <summary>
/// 插件所在目錄信息
/// </summary>
private static readonly DirectoryInfo PluginFolder;
/// <summary>
/// 程序應行時指定的dll目錄
/// </summary>
private static readonly DirectoryInfo ShadowCopyFolder;
public static void Initialize()
{
Directory.CreateDirectory(ShadowCopyFolder.FullName);
//清空插件dll運行目錄中的文件
foreach (var f in ShadowCopyFolder.GetFiles("*.dll", SearchOption.AllDirectories))
{
f.Delete();
}
foreach (var plug in PluginFolder.GetFiles("*.dll", SearchOption.AllDirectories).Where(i=>i.Directory.Parent.Name== "plugins"))
{
File.Copy(plug.FullName, Path.Combine(ShadowCopyFolder.FullName, plug.Name), true);
}
foreach (var a in
ShadowCopyFolder
.GetFiles("*.dll", SearchOption.AllDirectories)
.Select(x => AssemblyName.GetAssemblyName(x.FullName))
.Select(x => Assembly.Load(x.FullName)))
{
BuildManager.AddReferencedAssembly(a);
}
}
}
}
3.如何讓View引擎找到我們的視圖呢?答案是重寫RazorViewEngine的方法,我采用了約定大于配置的方式(假設我們的插件項目命名空間為Plugins.Apps.Sms,那么默認的控制器命名空間為Plugins.Apps.Sms.Controllers,插件生成后的文件夾必須為/Plugins/Plugins.Apps.Sms/),通過分析當前控制器就可以知道當前插件的View目錄位置
using System;
using System.Collections.Generic;
using System.Linq;
using System.Threading.Tasks;
using System.Web;
using System.Web.Mvc;
using System.Web.WebPages.Razor;
namespace Plugins.Web
{
public class CustomerViewEngine : RazorViewEngine
{
/// <summary>
/// 定義視圖頁所在地址。
/// </summary>
private string[] _viewLocationFormats = new[]
{
"~/Views/Parts/{0}.cshtml",
"~/Plugins/{pluginFolder}/Views/{1}/{0}.cshtml",
"~/Plugins/{pluginFolder}/Views/Shared/{0}.cshtml",
"~/Views/{1}/{0}.cshtml",
"~/Views/Shared/{0}.cshtml",
};
public override ViewEngineResult FindView(ControllerContext controllerContext, string viewName, string masterName, bool useCache)
{
string ns = controllerContext.Controller.GetType().Namespace;
string controller = controllerContext.Controller.GetType().Name.Replace("Controller", "");
//說明是插件中的控制器,View目錄需要單獨處理
if (ns.ToLower().Contains("plugins"))
{
var pluginsFolder = ns.ToLower().Replace(".controllers", "");
ViewLocationFormats = ReplacePlaceholder(pluginsFolder);
}
return base.FindView(controllerContext, viewName, masterName, useCache);
}
/// <summary>
/// 替換pluginFolder占位符
/// </summary>
/// <param name="folderName"></param>
private string[] ReplacePlaceholder(string folderName)
{
string[] tempArray = new string[_viewLocationFormats.Length];
if (_viewLocationFormats != null)
{
for (int i = 0; i < _viewLocationFormats.Length; i++)
{
tempArray[i] = _viewLocationFormats[i].Replace("{pluginFolder}", folderName);
}
}
return tempArray;
}
}
}
然后在主站點的Global.asax中將Razor引擎指定為我們重寫過的

4.開始制作一個插件目錄,跟我們平時建立的MVC項目并沒有太大區(qū)別,只是發(fā)布時需要做一些設置。
.生成路徑要按照第3條的約定來寫,不然會找不到視圖文件

.View目錄下的web.config和.cshtml文件要復制到生成目錄(在文件中點右鍵)


3.設置引用項目中的生成屬性,主程序下面已經(jīng)有了的就把“復制到輸出目錄”設置為無,要不然拷貝到動態(tài)bin目錄時會出錯,可以對第2步中的那個類改造一下,加入文件比較功能,bin目錄中沒有的,才拷貝到動態(tài)bin目錄中。

4.生成后的目錄結構如下:

5.跑一下,一切正常,插件中的控制器工作正常,視圖中引用了Model也沒問題

到此,一個插件系統(tǒng)的核心部分就算完成了,你可繼續(xù)進行擴展,增加插件的發(fā)現(xiàn)、安裝、卸載功能,這些相對于核心功能來說,都是小兒科。后續(xù)我會基于Abp框架出一個插件系統(tǒng)的文章,有興趣的把小板凳準備好,瓜子花生買上:)
五、源代碼
下載Plugins鏈接: https://pan.baidu.com/s/1nvmbL81 密碼: 85v1
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持腳本之家。
相關文章
asp.net下用url重寫URLReWriter實現(xiàn)任意二級域名的方法
asp.net下用url重寫URLReWriter實現(xiàn)任意二級域名的方法...2007-03-03
Asp.net中使用Sqlite數(shù)據(jù)庫的方法
Sqlite是最近比較流行的數(shù)據(jù)庫了,擁有比Access高效快速,易操作易實施。完全不需要在客戶端進行任何的配置,只需要在站點中引用入DLL文件即可使用了。2009-11-11
ASP.NET實現(xiàn)頁面?zhèn)髦档膸追N方法小結
這篇文章介紹了ASP.NET實現(xiàn)頁面?zhèn)髦档膸追N方法,有需要的朋友可以參考一下2013-11-11
asp.net下將Excel轉成XML檔的實現(xiàn)代碼
通過Asp.net(C#)應用程序讀取本地上傳的Excle文件,存放到DataSet中,通過DataSet中的方法直接生成XML文件.2009-11-11
在 .NET Core 中使用 Diagnostics (Diagnostic Source) 記錄跟蹤信息
今天給大家講一下在 .NET Core 2 中引入的全新 DiagnosticSource 事件機制,為什么說是全新呢? 在以前的 .NET Framework 有心的同學應該知道也有 Diagnostics,那么新的 .NET Core 中有什么變化呢?跟隨小編一起看看吧2021-06-06
ASP.Net Core基于ABP架構配置To Json序列化
這篇文章介紹了ASP.Net Core基于ABP架構配置To Json序列化的方法,文中通過示例代碼介紹的非常詳細。對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-06-06

