一起谈.NET技术,技巧:使用可扩展对象模式扩展HttpApplication

  概述

  HttpApplication对象对于做ASP.NET开发的朋友,我想没有人不熟悉它。在ASP.NET开发中,经常避免不了要在HttpApplication中执行一些操作,如使用了ASP.NET MVC框架,就会在Application_Start 事件中避免不了这样的路由规则配置代码:

protected void Application_Start(){RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

RouteTable.Routes.MapRoute("Default",                                              // Route name"{controller}/{action}/{id}",                           // URL with parametersnew { controller = "Home", action = "Index", id = "" }  // Parameter defaults);}

  如果仅仅是这一条,看起来倒不觉的有什么问题,但如果同时在应用程序中使用了工作流,又避免不了在Application_Start出现启动工作流运行时的代码:

protected void Application_Start(){// 注册路由规则RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");RouteTable.Routes.MapRoute("Default",                                              // Route name"{controller}/{action}/{id}",                           // URL with parametersnew { controller = "Home", action = "Index", id = "" }  // Parameter defaults);

// 启动工作流WorkflowRuntime workflowRuntime = new WorkflowRuntime("workflowServicesConfig");ExternalDataExchangeService     externalDataExchangeService = new ExternalDataExchangeService();    workflowRuntime.AddService(externalDataExchangeService);    workflowRuntime.StartRuntime();}

  试想一下,现在我们仅仅是有了ASP.NET MVC路由规则的配置、WF运行时的启动,如果在应用程序中使用某种DI框架,如微软的Unity,是不是又避免不了要出现这样的容器初始化代码呢?

protected void Application_Start(){// 注册路由规则RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");RouteTable.Routes.MapRoute("Default",                                              // Route name"{controller}/{action}/{id}",                           // URL with parametersnew { controller = "Home", action = "Index", id = "" }  // Parameter defaults);

// 启动工作流WorkflowRuntime workflowRuntime      = new WorkflowRuntime("workflowServicesConfig");ExternalDataExchangeService externalDataExchangeService       = new ExternalDataExchangeService();    workflowRuntime.AddService(externalDataExchangeService);    workflowRuntime.StartRuntime();

// 初始化DI容器IContainerContext repositoryContainer      = ContainerManager.GetContainer("repositoryContainer");    repositoryContainer.Initialize();}

  再看看Application_Start事件中的代码,有ASP.NET MVC的工作,有WF的工作,也有Unity的工作,不知道将来还会有什么?这些原本互相之间没有任何联系的代码,现在却同时堆在了一起,当每一部分(或者说每一个框架)变化的时候,都会涉及到Application_Start中代码的修改,显然违反了OCP原则。那么有没有一种机制,让这些互不相干的模块之间互相独立,各自发生变化时不影响对HttpApplication?此时我们就需要对HttpApplication进行扩展,提供一个扩展点,让其他模块的程序附加到HttpApplication上面。

  可扩展对象模式

  我们知道WCF提供了非常完美的扩展机制,几乎在服务执行过程中的每一个环节上都提供有扩展点,如ServiceHostBase、OperationContext、InstanceContext、IContextChannel,这些对象都属于可扩展对象,它们都通过Extensions属性获取用于所有扩展的集合。我们能不能使用这种方式对HttpApplication也进行扩展呢,答案自然是肯定的。查阅一下MSDN就会知道在System.ServiceModel命名空间下面提供了这样的一组接口:IExtensibleObject、IExtension和IExtensionCollection,这是可扩展对象模式中最重要的三个接口,也只有这三个接口。

  IExtensibleObject自然是定义了可扩展对象,即我们要对谁进行扩展,它的定义非常简单,仅仅是提供了一个只读的属性Extensions,用来提供所有扩展对象的集合,如下代码所示:

public interface IExtensibleObject<T> where T : IExtensibleObject<T>{IExtensionCollection<T> Extensions { get; }}

  IExtension定义了扩展对象的契约,使对象可以通过聚合扩展另一个对象(此处的另一个对象,就是指上面所讲的扩展宿主IExtensibleObject),在IExtension中定义了两个非常重要的方法Attach和Detach方法,分别用来提供聚合或解聚的通知。

public interface IExtension<T> where T : IExtensibleObject<T>{void Attach(T owner);void Detach(T owner);}

  当一个扩展对象IExtension附加到可扩展对象的扩展集合中时,它的Attach方法将会被调用;反之如果从集合中移除一个扩展对象时,它的Detach方法会被调用。这一点我们可以通过Reflector来得到验证,如下代码所示:

protected override void InsertItem(int index, IExtension<T> item){lock (base.SyncRoot)    {        item.Attach(this.owner);base.InsertItem(index, item);    }}

protected override void RemoveItem(int index){lock (base.SyncRoot)    {base.Items[index].Detach(this.owner);base.RemoveItem(index);    }}

  最后一个接口是IExtensionCollection,它是IExtension对象的集合。

  对HttpApplication进行扩展

  下面我们就看一下如何使用可扩展对象模式对HttpApplication进行扩展,首先定义可扩展对象,让ExtensibleHttpApplication派生于HttpApplication,并实现了IExtensibleObject接口,泛型的参数类型就是它自身,如下代码所示:

public class ExtensibleHttpApplication : HttpApplication,IExtensibleObject<ExtensibleHttpApplication>{private IExtensionCollection<ExtensibleHttpApplication> _extensions;

public ExtensibleHttpApplication()    {this._extensions = new ExtensionCollection<ExtensibleHttpApplication>(this);    }

public IExtensionCollection<ExtensibleHttpApplication> Extensions    {get{return this._extensions;        }    }}

  有了可扩展的HttpApplication之后,需要在HttpApplication中实现任何功能,都可以作为一个扩展附加到ExtensibleHttpApplication上去,如实现ASP.NET MVC路由,可以定义一个如下代码所示的扩展对象:

public class MvcHttpApplication : IExtension<ExtensibleHttpApplication>{public void Attach(ExtensibleHttpApplication owner)    {RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

RouteTable.Routes.MapRoute("Default",                                              // Route name"{controller}/{action}/{id}",                           // URL with parametersnew { controller = "Home", action = "Index", id = "" }  // Parameter defaults);    }

public void Detach(ExtensibleHttpApplication owner)    {//nothing}}

  同样如果要在HttpApplication中启动Workflow,可以再针对Workflow定义一个扩展对象,如下示例代码所示:

public class WorkflowHttpApplication : IExtension<ExtensibleHttpApplication>{private WorkflowRuntime workflowRuntime;public void Attach(ExtensibleHttpApplication owner)    {        workflowRuntime = new WorkflowRuntime("workflowServicesConfig");ExternalDataExchangeService externalDataExchangeService        = new ExternalDataExchangeService();        workflowRuntime.AddService(externalDataExchangeService);        workflowRuntime.StartRuntime();    }

public void Detach(ExtensibleHttpApplication owner)    {        workflowRuntime.StopRuntime();    }}

  我们已经定义好了相应的扩展对象,只需要在相应的HttpApplication把扩展对象附加到ExtensibleHttpApplication上即可,修改Global.asax中的代码如下所示:

public class MvcApplication : ExtensibleHttpApplication{protected void Application_Start()    {this.Extensions.Add(new MvcHttpApplication());this.Extensions.Add(new WorkflowHttpApplication());    }}

  现在代码是不是看起来优雅多了?现在如果要在Application_Start中,添加另外一些执行代码,只需要编写相应的扩展对象,并将其添加到Extension集合中即可。也许有朋友会问,这样每添加一些新的代码,还是要修改Application_Start中的代码啊?别忘了,可以通过配置可以解决这个问题,WCF中的扩展不也是可以采用配置方式实现,不是吗?同样,如果我们需要在Application_End事件中释放某些对象,可以直接从扩展集合中移除它,此时将会调用它的Detach方法。

  总结

  本文介绍了如何使用WCF中提供的可扩展对象模式扩展HttpApplication,事实上可扩展对象模式的作用远不在此,它可以扩展.NET类库中任何我们想对其进行扩展的对象,或者是一个自定义的类型,都可以使用可扩展对象模式对其进行扩展。

时间: 2024-07-30 09:59:06

一起谈.NET技术,技巧:使用可扩展对象模式扩展HttpApplication的相关文章

技巧:使用可扩展对象模式扩展HttpApplication

概述 HttpApplication对象对于做ASP.NET开发的朋友,我想没有人不熟悉它.在ASP.NET开发中,经常避免不了要在HttpApplication中执行一些操作,如使用了ASP.NET MVC框架,就会在Application_Start 事件中避免不了这样的路由规则配置代码: protected void Application_Start() { RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}");

使用可扩展对象模式扩展HttpApplication

概述 HttpApplication对象对于做ASP.NET开发的朋友,我想没有人不熟悉它.在ASP.NET开发中,经常避免 不了要在HttpApplication中执行一些操作,如使用了ASP.NET MVC框架,就会在Application_Start 事件 中避免不了这样的路由规则配置代码: protected void Application_Start() { RouteTable.Routes.IgnoreRoute("{resource}.axd/{*pathInfo}"

来源于WCF的设计模式:可扩展对象模式[上篇]

我一直很喜欢剖析微软一些产品.框架的底层实现.在我看来,这不但让我们可以更加深入地了解其运作的原理,同时也能提高设计/架构的技能.因为对于这些框架或者产品来说,高质量的设计是它们能够成功的一个最基本的因素.比如说比如ASP.NET,不但能够支持传统的Web Form应用,MVC同样建立在它基础之上.比如说WCF,从其诞生的那一天开始,真个架构体系就从未改变.这些应用在这些产品和框架上的设计其实是最值得我们学习的设计案例.比如说,今天我们介绍的"可扩展对象模式(Extensible Object

来源于WCF的设计模式:可扩展对象模式[下篇]

在<来源于WCF的设计模式:可扩展对象模式>我通过一个简单的例子介绍了基于IExtensibleObject<T>和IExtension<T>这两个接口为核心的"可扩展对象模式".在那篇文章中,我是通过编程的方式来应用扩展到扩展对象的.其实,如何能够通过配置的方式来定义扩展,这个所谓的"可扩展对象模式"将会发挥更大的威力.[源代码从这里下载] 目录: 一.将XxxBuilder定义在配置中 二.ExtensionConfigura

一起谈.NET技术,XML与DataSet对象的关系

在.NET Framework 中,经常使用XML 作为存储和传输各种数据的格式.DataSet 中的数据可以转换成XML 的形式来表示和存储.我们可以使用XML 对象同步和转换DataSet 的数据,而DataSet 也可以存储和传输XML 格式的数据. XML 与 DataSet 的关系如下图所示: DataSet 对象的常用方法如下: A.  使用ReadXml( ) 方法:从文件或流中加载XML 数据,填充DataSet 对象.DataSet 对象.ReadXML( 文件路径字符串|st

一起谈.NET技术,由扩展方法引申出的编程思维

1. Helper大爆炸 .NET Framework为我们提供了丰富的类库,但是这并不是万能地,在大部分的时间,我们都需要为我们的项目特殊定制我们的通用类库. 常常,我们都可以构造一个类,类里封装一些方法.但是对于很多时候,我们并没有办法提取出这样一个类,举一个小例子,我们在很多时候,需要把url给保存到数据库里,作为一个唯一标识,但是我们知道url所占空间很大,如果用url来建立索引的话是非常耗费空间,而且影响效率的,那么我们最常用的办法就是把url做一个Hash来作为索引的替代品. 这个时

一起谈.NET技术,从扩展方法到流畅的程序体验(一)

今天让公司的程序员试用了一下还在开发中的代号为"Jumony"的HTML数据绑定引擎,开发人员的一句评价被我视为最高的褒奖. "感觉这个框架就是,你想到什么就写什么." 想到什么就写什么,在这个越来越强调快速开发的时代,这一点变得越来越重要.我最近经常戏言:"natural code才是王道",当然,不是说我们要用中文去编程,而是程序应该成为越来越自然的表达. 让程序员获得流畅的编程体验,是将来每一个框架都必须去考虑和实现的事情.随着.NET F

一起谈.NET技术,分享一些非常好用的Visual Studio扩展

扩展管理器 Visual Studio 2010新增"扩展管理器"的功能,可用于添加.移除.启用和禁用Visual Studio扩展.我们可以从Visual Studio Gallery(Visual Studio库)网站上安装扩展,这些扩展一般是VSIX包格式,包括项目模板.项模板.工具箱项.托管扩展框架(MEF)组件和VSPackage.安装在"Visual Studio安装文件夹\Common7\IDE\Extensions\Company\Product\Versio

linux平台编译安装PHP7并安装Redis扩展与Swoole扩展实例教程_php技巧

本文实例讲述了linux平台编译安装PHP7并安装Redis扩展与Swoole扩展的方法.分享给大家供大家参考,具体如下: 前面<PHP7安装Redis扩展教程[Linux与Windows平台]>一文告诉读者简单的安装Redis的方法,下面我们来实现在linux中编译安装PHP7并安装Redis扩展与Swoole扩展的方法. 编译安装PHP7并安装Redis扩展Swoole扩展: 在编译php7的机器上已经有编译安装过php5.3以上的版本,从而依赖库都有了 本php7是编译成fpm-php