一,AOP(Aspect Oriented Programming)
在谈AOP之前,我们先来说OOP(Object Oriented Programming),这是接触每一门编程语言的时候,都会先学的东西。众所周知,OOP有三个特征:封装,继承,多态。这是从类的角度来说的,如果我们把系统中的关系用UML图来表示的话,会发现有点儿类似树的结构,这棵树是从上到下的或者是从下到上的,是一个竖直的结构。
但是AOP不同,AOP是一个水平的结构。在一个系统中,我们可能会遇到这样的情况,当我们写了一大段业务逻辑,之后,在方法结尾,我们要将一些操作记入日志中,然而,我们要记入日志的东西很多,很可能我们很多代码都是这个结构:业务逻辑+日志记录。为解耦业务逻辑跟这些与业务逻辑无关的东西,方便系统维护,提高代码清晰度,我们加入AOP。
如果说OOP是对系统的纵切的话,那么AOP就是对系统的横切,在横切的同时,加入统一的处理,让我们编码的时候,重点还是放在业务逻辑方面,对系统整体的相似的东西(与业务逻辑无关的东西)在横切的同时统一加入。
二,Filter实现简单AOP
在ASP.NET MVC框架中,为我们提供了四种类型的Filter类型包括:IAuthorizationFilter、IActionFilter、IResultFilter、IExceptionFilter,执行顺序如下:
IAuthorizationFilter 最先执行的,在这些Filter中,我们可以说它的执行优先级是最高的,用于身份验证并对控制器中的action进行授等进行逻辑处理
IActionFilter 在IAuthorizationFilter之后执行,包含两个方法,在controller中的action执行之前、执行之后进行逻辑处理
IResultFilter 同样包含两个方法,在IActionFilter之后执行,在返回View之前和返回View之后执行逻辑处理
IException 主要用于对异常信息进行处理
下面我们以IActionFilter与IException为例,来看下filter是如何实现AOP的。
首先是actionFilter:
public class LoggerFilter:FilterAttribute,IActionFilter
{
/// <summary>
/// 在被拦截action后执行
/// </summary>
/// <param name="filterContext">The filter context.</param>
/// <remarks>创建者:刘慧超; 创建时间:2015-09-13 19:47:23</remarks>
public void OnActionExecuted(ActionExecutedContext filterContext)
{
filterContext.Controller.ViewData["executedLogger"] = "action之后。。——" + DateTime.Now;
}
/// <summary>
/// 在被拦截Action前执行
/// </summary>
/// <param name="filterContext">The filter context.</param>
/// <remarks>创建者:刘慧超; 创建时间:2015-09-13 19:47:00</remarks>
public void OnActionExecuting(ActionExecutingContext filterContext)
{
filterContext.Controller.ViewData["executingLogger"] = "action之前。。——" + DateTime.Now;
}
}
我们建立的类要继承FilterAttribute,实现IActionFilter,重写action执行之前和之后的方法。
测试,如果要使用我们的filter,只需在返回view的方法上面加上标签即可。
[LoggerFilter()]
public ActionResult Index()
{
System.Threading.Thread.Sleep(2000);
ViewData["Time"] = DateTime.Now;//当前时间
System.Threading.Thread.Sleep(2000);
return View();
}
然后我们在页面上接收下根据打印的时间来判断执行顺序:
@using System.Web.UI.WebControls
@{
ViewBag.Title = "Index";
}
<h2>测试ActionFilter</h2>
<p>@ViewData["executingLogger"]</p>
<p></p>
<p>@ViewData["Time"]</p>
<p></p>
<p>@ViewData["executedLogger"]</p>
异常处理:
public class ExceptionFilter:FilterAttribute,IExceptionFilter
{
/// <summary>
/// 异常发生时候被调用
/// </summary>
/// <param name="filterContext">The filter context.</param>
/// <remarks>创建者:刘慧超; 创建时间:2015-09-13 19:52:25</remarks>
public void OnException(ExceptionContext filterContext)
{
filterContext.Controller.ViewData["ErrorMessage"] = filterContext.Exception.Message; //此处Exception是异常类型实例
filterContext.Result = new ViewResult()
{
ViewName = "ExceptionIndex", //出错后跳转到的错误页面
ViewData = filterContext.Controller.ViewData
};
filterContext.ExceptionHandled = true; //告诉系统,异常已经处理了,不需要再次处理了
}
}
在代码中,我们定义出现异常之后,跳转到异常出错页面。
测试方法:
public void TestE()
{
throw new Exception("这是个被抛出的异常,啦啦啦。。。");
return;
;
}
[ExceptionFilter()]
public ActionResult ExceptionIndex()
{
TestE();
return View();
}
页面显示出一次信息:
@{
ViewBag.Title = "ExceptionIndex";
}
<h2>异常拦截测试</h2>
<p>@ViewData["ErrorMessage"]</p>
可见filter使用方法类似,首先定义继承相应接口和特性标签的类,之后在要处理的返回视图的方法上加上attribute.
有点儿类似我们以前经常被问道的页面生命周期,但是又不一样,页面生命周期是类自带的,而filter中方法的执行是后期有选择的加入的,感觉很像方法正在从上到下流畅执行,到了特定地方,被横切一刀,切面编程还是蛮形象的。