net搭建热插拔式web框架(沙箱的构建)
net搭建热插拔式web框架(沙箱的构建)
上周五写了一个实现原理篇,在评论中看到有朋友也遇到了我的问题,真的是有种他乡遇知己的感觉,整个系列我一定会坚持写完,并在最后把代码开源到git中。上一篇文章很多人看了以后,都表示不解,觉得不知道我到底要干什么,可能就像隔行如隔山吧,就像做移动端开发的人很少去考虑分布式中的通信一样。大家都知道模块化,但模块化的思路有很多,我的只是其中一种,也许你看到最后会觉得这种思路在经过不断地演化后会成为一种很好的解决方案,当然这离不开以后大家对代码及思想的贡献。
好了不扯了,还是回到主题上来吧....
沙箱是什么?怎么用呢?
沙箱说白了就是插件、模块运行的环境,有点像docker又不像(哈哈)。沙箱主要由两部分组成:沙箱管道和沙箱启动器(为了显得高大上一些,就起了两个难以理解的名字),还有一个以后为功能做铺垫的实体类——controller/action 封包类
首先贴一下这个实体类的代码:

/// <summary>controller/action 封包类
/// </summary>
public class CAModel
{
public string ControllerName{get;set;}
public string ActionName{get;set;}
public string UrlPath { get; set; }
/// <summary>构造
/// </summary>
/// <param name="controllerName">controller 全名(带命名空间)</param>
/// <param name="actionName">action 全名(不带参数)</param>
public CAModel(string controllerName,string actionName)
{
ControllerName=controllerName;
ActionName=actionName;
UrlPath = controllerName.Replace(".Areas.", "/").Replace(".Controllers.", "/");//controller转Url
if (UrlPath.EndsWith("Controller"))
{
UrlPath = string.Format("/{0}/{1}", UrlPath.Substring(0, UrlPath.Length - 10),actionName);
}
} public CAModel()
{
}
}

这个封装类主要是为了以后系统核心功能:权限管理,方便获取模块内所有action对应的Url路径,现在就不过多投入精力了。
下边我们说一下沙箱管道(SandBoxChannel ):沙箱管道服务于沙箱启动器,这个类需要继承MarshalByRefObject,也就是必须要支持跨域访问。这个类中最关键的就是_assembly和InvokeMothod。
/// <summary>沙箱管道
/// </summary> public class SandBoxChannel : MarshalByRefObject { /// <summary>沙箱内加载的程序集 /// </summary> private Assembly _assembly; /// <summary>加载程序集 /// </summary> /// <param name="assemblyFile">程序集文件路径(主程序类库路径,依赖类库自动加载)</param> public void LoadAssembly(string assemblyFile) { try { _assembly = Assembly.LoadFrom(assemblyFile); } catch (Exception ex) { throw ex; } } /// <summary>沙箱内方法调用 /// </summary> /// <param name="typeName">类名称(全名称)</param> /// <param name="methodName">方法名称</param> /// <param name="args">参数</param> /// <returns></returns> public object InvokeMothod(string typeName, string methodName, params object[] args) { var _assembly_temp = _assembly; if (_assembly_temp == null) return null; if (typeName == "Huber.Kernel.Entity.CurVariable" && methodName == "SetCurWebDir") {//设置当前沙箱内的系统变量,非外部方法 foreach (var a in AppDomain.CurrentDomain.GetAssemblies()) { if (a.GetName().Name == "Huber.Kernel") { _assembly_temp = a; break; } } } Type tp = _assembly_temp.GetType(typeName, true, false); if (tp == null) return null; MethodInfo method = tp.GetMethod(methodName); if (method == null) return null; Object obj = Activator.CreateInstance(tp); //ContentResult //FileContentResult //FileStreamResult //FilePathResult //HttpNotFoundResult //JavaScriptResult //JsonResult //PartialViewResult //RedirectToRouteResult //RedirectResult //ViewResult object result = method.Invoke(obj, args); return result; } /// <summary>获取所有的action /// </summary> /// <returns></returns> public Dictionary<string, CAModel> GetAllAction() { Dictionary<string, CAModel> result = null; Type[] types = _assembly.GetTypes(); MethodInfo[] meths = null; string controller = string.Empty; if (types != null) { result = new Dictionary<string, CAModel>(); string url = string.Empty; CAModel temp; foreach (var t in types) { if (t.BaseType != null && t.BaseType.ToString() == "Huber.Kernel.MVC.HuberController") { meths = t.GetMethods(BindingFlags.Instance | BindingFlags.Public | BindingFlags.DeclaredOnly); foreach (var m in meths) { temp = new CAModel(t.ToString(), m.Name); result.Add(temp.UrlPath.ToLower(), temp); } } } } return result; } public override object InitializeLifetimeService() { //Remoting对象 无限生存期 return null; }再看一下沙箱启动器(SandBoxDynamicLoader):所谓沙箱启动器就是创建一个启动器对象,把模块的类库、配置文件等加载进去,当然这个启动内部是一个沙箱(即appdomain)。在SandBoxDynamicLoader的构造方法中创建的了一个appdomain和一个SandBoxChannel对象,支持模块的加载与卸载
public class SandBoxDynamicLoader
{ /// <summary>沙箱对应的应用程序域 /// </summary> private AppDomain appDomain; /// <summary>沙箱管道 /// </summary> private SandBoxChannel channelChannel; /// <summary>程序域的ID /// </summary> public int AppDomainID { get; set; } /// <summary>插件名称 /// </summary> public string PluginName { get; set; } /// <summary>构造 /// </summary> /// <param name="ApplicationBase">插件的所在的目录(bin目录)</param> /// <param name="_PluginName">插件名称</param> /// <param name="configPath">config文件位置</param> /// <param name="_AppDomainID">域标识(唯一)</param> public SandBoxDynamicLoader(string ApplicationBase, string _PluginName, string configPath, int _AppDomainID) { PluginName = _PluginName; AppDomainID = _AppDomainID; AppDomainSetup setup = new AppDomainSetup(); setup.ApplicationBase = ApplicationBase; DirectoryInfo di=new DirectoryInfo(ApplicationBase); if (configPath != string.Empty) { setup.ConfigurationFile = configPath; } setup.PrivateBinPath = Path.Combine(AppDomain.CurrentDomain.BaseDirectory, "private"); setup.CachePath = setup.ApplicationBase; setup.ShadowCopyFiles = "true"; setup.ShadowCopyDirectories = setup.ApplicationBase+"\\SandBoxRunShadow"; AppDomain.CurrentDomain.SetShadowCopyFiles(); this.appDomain = AppDomain.CreateDomain(PluginName, null, setup); this.appDomain.SetData("APP_CONFIG_FILE", configPath); String name = Assembly.GetExecutingAssembly().GetName().FullName; try { this.channelChannel = (SandBoxChannel)this.appDomain.CreateInstanceAndUnwrap(name, typeof(SandBoxChannel).FullName); } catch (Exception ex) { } } /// <summary>加载程序集 /// </summary> /// <param name="assemblyFile"></param> public void LoadAssembly(string assemblyFile) { channelChannel.LoadAssembly(assemblyFile); } /// <summary>获取当前模块内所有action /// </summary> /// <returns></returns> public Dictionary<string, CAModel> GetAllAction() { if (channelChannel == null) return null; return channelChannel.GetAllAction(); } /// <summary>方法调用 /// </summary> /// <param name="typeName">类名称(全名称)</param> /// <param name="methodName">方法名称</param> /// <param name="args">参数</param> /// <returns></returns> public object InvokeMothod(string typeName, string methodName, params object[] args) { return channelChannel.InvokeMothod(typeName, methodName, args); } /// <summary>卸载 /// </summary> public void Unload() { try { if (appDomain == null) return; AppDomain.Unload(this.appDomain); this.appDomain = null; } catch (CannotUnloadAppDomainException ex) { throw ex; } }到此沙箱模型就完了,其实整个过程可以归纳为:创建一个appdomain,利用反射调用方法处理请求。这个模型不仅在web平台上可以使用,其实他早就在系统服务型框架、窗体框架中大范围使用了。
转载请注明出处:http://www.cnblogs.com/eric-z/p/5028243.html
net搭建热插拔式web框架(沙箱的构建)的更多相关文章
- 第二篇 基于.net搭建热插拔式web框架(沙箱的构建)
上周五写了一个实现原理篇,在评论中看到有朋友也遇到了我的问题,真的是有种他乡遇知己的感觉,整个系列我一定会坚持写完,并在最后把代码开源到git中.上一篇文章很多人看了以后,都表示不解,觉得不知道我到底 ...
- 第三篇 基于.net搭建热插拔式web框架(重造Controller)
由于.net MVC 的controller 依赖于HttpContext,而我们在上一篇中的沙箱模式已经把一次http请求转换为反射调用,并且http上下文不支持跨域,所以我们要重造一个contro ...
- 基于.net搭建热插拔式web框架(实现原理)
第一节:我们为什么需要一个热插拔式的web框架? 模块之间独立开发 假设我们要做一个后台管理系统,其中包括“用户活跃度”.“产品管理”."账单管理"等模块.每个模块中有自己的业务特 ...
- net搭建热插拔式web框架
net搭建热插拔式web框架(重造Controller) 由于.net MVC 的controller 依赖于HttpContext,而我们在上一篇中的沙箱模式已经把一次http请求转换为反射调用,并 ...
- 第五篇 基于.net搭建热插拔式web框架(拦截器---请求管道)
好了,前边我们把核心内容介绍完了,接下来要做的就是拦截用户的请求,并把请求转向沙箱内. 这里我们准备通过实现一个HttpModule类来完成请求的拦截与转发.新建一个HuberHttpModule类, ...
- 第四篇 基于.net搭建热插拔式web框架(RazorEngine实现)
在开头也是先给大家道个歉,由于最近准备婚事导致这篇文章耽误了许久,同时也谢谢老婆大人对我的支持. 回顾上篇文章,我们重造了一个controller,这个controller中用到了视图引擎,我们的视图 ...
- 架构探险——第三章(搭建轻量级Java Web框架)
解决的问题 servlet的数量会随业务功能的扩展而不断增加,我们有必要减少servlet的数量,交给controller处理,它负责调用service的相关方法,并将返回值放入request或res ...
- 读《架构探险——从零开始写Java Web框架》
内容提要 <架构探险--从零开始写Java Web框架>首先从一个简单的 Web 应用开始,让读者学会如何使用 IDEA.Maven.Git 等开发工具搭建 Java Web 应用:接着通 ...
- 什么是web框架?
英文原文:http://jeffknupp.com/blog/2014/03/03/what-is-a-web-framework/ 在原文基础上加上了自己在翻译过程中,查看的资料和自己的一些理解,同 ...
随机推荐
- poj3642 Charm Bracelet(0-1背包)
题目意思: 给出N,M,N表示有N个物品,M表示背包的容量.接着给出每一个物品的体积和价值,求背包可以装在的最大价值. http://poj.org/problem? id=3624 题目分析: o- ...
- 【iOS开发】 常遇到的Crash和Bug处理
一,Unknown type name .... 如果是报这个错误,多半是你的对象类型没有被识别,检查是不是没有引用对应的库或者头文件在你的文件头部分,还有可能是循环引用导致的,循环引用的解决方法就是 ...
- UVa 1292 - Strategic game (树形dp)
本文出自 http://blog.csdn.net/shuangde800 题目链接: 点击打开链接 题目大意 给定一棵树,选择尽量少的节点,使得每个没有选中的结点至少和一个已选结点相邻. 思路 ...
- cocos2d_x_05_Box2D物理引擎
一.认识Box2D 帮助文档,共69页 二.创建一个物理世界 先导入主头文件 #include <Box2D/Box2D.h> 三.物理世界一览 像素转成米 的比例因子 就是32 三.运动 ...
- 采用Bash脚本性能监控过程
为一个Linux过程监控,采用Bash脚本. 采用ps命令的过程监控,使用周期加上连续监测的睡眠时间. 使用方法: psmonitor.sh -p [pid] -d [interval] -n [st ...
- Velocity缓存与穿透(转)
原文 http://fantaxy025025.iteye.com/blog/2283904 主题 Java 总评:只是标记一下这种用法hack方法.这种场景下用这种方法还是比较雷人的. Veloc ...
- dpdk组态 千兆网卡 驱动 失败 原因分析及 解决方案
dpdk版本号是1.7.1稳定版,server它是ubuntu12.04LTS x86 64bit 绑定默认驱动程序千兆网卡ixgbe失败 # ./dpdk_nic_bind.py -b ixgbe ...
- poj 2253 Frogger (最长路中的最短路)
链接:poj 2253 题意:给出青蛙A,B和若干石头的坐标,现青蛙A想到青蛙B那,A可通过随意石头到达B, 问从A到B多条路径中的最长边中的最短距离 分析:这题是最短路的变形,曾经求的是路径总长的最 ...
- HTML5 CSS3 诱人的实例 : 网页载入进度条的实现,下载进度条等
今天给大家带来一个比較炫的进度条,进度条在一耗时操作上给用户一个比較好的体验,不会让用户认为在盲目等待,对于没有进度条的长时间等待,用户会任务死机了,毫不犹豫的关掉应用:一般用于下载任务,删除大量任务 ...
- 阅读:AirBag Boosting Smartphone Resistance to Malware Infection