C#进阶系列——AOP?AOP!

前言:这篇打算写写aop,说起aop,其实博主接触这个概念也才几个月,了解后才知道,原来之前自己写的好多代码原理就是基于aop的,比如mvc的过滤器filter,它里面的异常捕捉可以通过filterattribute,iexceptionfilter去处理,这两个对象的处理机制内部原理应该就是aop,只不过之前没有这个概念罢了。

一、AOP概念

老规矩,还是先看官方解释:AOP(Aspect-Oriented Programming,面向切面的编程),它是可以通过预编译方式和运行期动态代理实现在不修改源代码的情况下给程序动态统一添加功能的一种技术。它是一种新的方法论,它是对传统OOP编程的一种补充。OOP是关注将需求功能划分为不同的并且相对独立,封装良好的类,并让它们有着属于自己的行为,依靠继承和多态等来定义彼此的关系;AOP是希望能够将通用需求功能从不相关的类当中分离出来,能够使得很多类共享一个行为,一旦发生变化,不必修改很多类,而只需要修改这个行为即可。AOP是使用切面(aspect)将横切关注点模块化,OOP是使用类将状态和行为模块化。在OOP的世界中,程序都是通过类和接口组织的,使用它们实现程序的核心业务逻辑是十分合适。但是对于实现横切关注点(跨越应用程序多个模块的功能需求)则十分吃力,比如日志记录,权限验证,异常拦截等。

博主的理解:AOP就是将公用功能提取出来,如果以后公用功能的需求发生变化,只需要改动公用的模块的代码即可,多个调用的地方则不需要改动。所谓面向切面,就是只关注通用功能,而不关注业务逻辑。实现方式一般是通过拦截。比如,我们随便一个Web项目基本都有的权限验证功能,进入每个页面前都会校验当前登录用户是否有权限查看该界面,我们不可能说在每个页面的初始化方法里面都去写这段验证的代码,这个时候我们的AOP就派上用场了,AOP的机制是预先定义一组特性,使它具有拦截方法的功能,可以让你在执行方法之前和之后做你想做的业务,而我们使用的时候只需要的对应的方法或者类定义上面加上某一个特性就好了。

二、使用AOP的优势

博主觉得它的优势主要表现在:

1、将通用功能从业务逻辑中抽离出来,可以省略大量重复代码,有利于代码的操作和维护。

2、在软件设计时,抽出通用功能(切面),有利于软件设计的模块化,降低软件架构的复杂度。也就是说通用的功能都是一个单独的模块,在项目的主业务里面是看不到这些通用功能的设计代码的。

三、AOP的简单应用

为了说明AOP的工作原理,博主打算先从一个简单的例子开始,通过静态拦截的方式来了解AOP是如何工作的。

1、静态拦截

public class Order{    public int Id { set; get; }    public string Name { set; get; }    public int Count { set; get; }    public double Price { set; get; }    public string Desc { set; get; }} public interface IOrderProcessor{    void Submit(Order order);}public class OrderProcessor : IOrderProcessor{    public void Submit(Order order)    {        Console.WriteLine("提交订单");    }} public class OrderProcessorDecorator : IOrderProcessor{    public IOrderProcessor OrderProcessor { get; set; }    public OrderProcessorDecorator(IOrderProcessor orderprocessor)    {        OrderProcessor = orderprocessor;    }    public void Submit(Order order)    {        PreProceed(order);        OrderProcessor.Submit(order);        PostProceed(order);    }    public void PreProceed(Order order)    {        Console.WriteLine("提交订单前,进行订单数据校验....");        if (order.Price 0)        {            Console.WriteLine("订单总价有误,请重新核对订单。");        }    }     public void PostProceed(Order order)    {        Console.WriteLine("提交带单后,进行订单日志记录......");        Console.WriteLine(DateTime.Now.ToString("yyyy-MM-dd HH:mm:ss") + "提交订单,订单名称:" + order.Name + ",订单价格:" + order.Price);    }}

调用代码:

static void Main(string[] args)   {       Order order = new Order() { Id = 1, Name = "lee", Count = 10, Price = 100.00, Desc = "订单测试" };       IOrderProcessor orderprocessor = new OrderProcessorDecorator(new OrderProcessor());       orderprocessor.Submit(order);       Console.ReadLine();   }

得到结果:

969.jpg

上面我们模拟订单提交的例子,在提交一个订单前,我们需要做很多的准备工作,比如数据有效性校验等;订单提交完成之后,我们还需要做日志记录等。上面的代码很简单,没有任何复杂的逻辑,从上面的代码可以看出,我们通过静态植入的方式手动在执行方法前和执行方法后让它做一些我们需要的功能。AOP的实现原理应该也是如此,只不过它帮助我们做了方法拦截,帮我们省去了大量重复代码,我们要做的仅仅是写好拦截前和拦截后需要处理的逻辑。

2、动态代理

了解了静态拦截的例子,你是否对AOP有一个初步的认识了呢。下面我们就来到底AOP该如何使用。按照园子里面很多牛人的说法,AOP的实现方式大致可以分为两类:动态代理和IL 编织两种方式。博主也不打算照本宣科,分别拿Demo来说话吧。下面就以两种方式各选一个代表框架来说明。

动态代理方式,博主就以微软企业库(MS Enterprise Library)里面的PIAB(Policy Injection Application Block)框架来作说明。

首先需要下载以下几个dll,然后添加它们的引用。

970.jpg

然后定义对应的Handler

public class User {     public string Name { set; get; }     public string PassWord { set; get; } }  #region 1、定义特性方便使用 public class LogHandlerAttribute : HandlerAttribute {     public string LogInfo { set; get; }     public int Order { get; set; }     public override ICallHandler CreateHandler(IUnityContainer container)     {         return new LogHandler() { Order = this.Order, LogInfo = this.LogInfo };     } } #endregion  #region 2、注册对需要的Handler拦截请求 public class LogHandler : ICallHandler {     public int Order { get; set; }     public string LogInfo { set; get; }      //这个方法就是拦截的方法,可以规定在执行方法之前和之后的拦截     public IMethodReturn Invoke(IMethodInvocation input, GetNextHandlerDelegate getNext)     {         Console.WriteLine("LogInfo内容" + LogInfo);         //0.解析参数         var arrInputs = input.Inputs;         if (arrInputs.Count > 0)         {             var oUserTest1 = arrInputs[0] as User;         }         //1.执行方法之前的拦截         Console.WriteLine("方法执行前拦截到了");         //2.执行方法         var messagereturn = getNext()(input, getNext);          //3.执行方法之后的拦截         Console.WriteLine("方法执行后拦截到了");         return messagereturn;     } } #endregion  #region 3、用户定义接口和实现 public interface IUserOperation {     void Test(User oUser);     void Test2(User oUser, User oUser2); }  //这里必须要继承这个类MarshalByRefObject,否则报错 public class UserOperation : MarshalByRefObject, IUserOperation {     private static UserOperation oUserOpertion = null;     public UserOperation()     {         //oUserOpertion = PolicyInjection.Create();     }      //定义单例模式将PolicyInjection.Create()产生的这个对象传出去,这样就避免了在调用处写这些东西     public static UserOperation GetInstance()     {         if (oUserOpertion == null)             oUserOpertion = PolicyInjection.Create();          return oUserOpertion;     }     //调用属性也会拦截     public string Name { set; get; }      //[LogHandler],在方法上面加这个特性,只对此方法拦截     [LogHandler(LogInfo = "Test的日志为aaaaa")]     public void Test(User oUser)     {         Console.WriteLine("Test方法执行了");     }      [LogHandler(LogInfo = "Test2的日志为bbbbb")]     public void Test2(User oUser, User oUser2)     {         Console.WriteLine("Test2方法执行了");     } } #endregion

最后我们来看调用的代码:

static void Main(string[] args){    try    {        var oUserTest1 = new User() { Name = "test2222", PassWord = "yxj" };        var oUserTest2 = new User() { Name = "test3333", PassWord = "yxj" };        var oUser = UserOperation.GetInstance();        oUser.Test(oUserTest1);        oUser.Test2(oUserTest1,oUserTest2);    }    catch (Exception ex)    {        //throw;    }}

得到结果如下:

971.jpg

我们来看执行Test()方法和Test2()方法时候的顺序。

972.jpg

由于Test()和Test2()方法上面加了LogHander特性,这个特性里面定义了AOP的Handler,在执行Test和Test2方法之前和之后都会进入Invoke()方法里面。其实这就是AOP的意义所在,将切面的通用功能在统一的地方处理,在主要逻辑里面直接用过特性使用即可。

3、IL编织

静态织入的方式博主打算使用PostSharp来说明,一来这个使用起来简单,二来项目中用过这种方式。

Postsharp从2.0版本就开始收费了。为了说明AOP的功能,博主下载了一个免费版本的安装包,使用PostSharp与其它框架不太一样的是一定要下载安装包安装,只引用类库是不行的,因为上文说过,AOP框架需要为编译器或运行时添加扩展。使用步骤如下:

(1)下载Postsharp安装包,安装。

(2)在需要使用AOP的项目中添加PostSharp.dll 这个dll的引用。

(3)定义拦截的方法:

[Serializable]public class TestAop : PostSharp.Aspects.OnMethodBoundaryAspect{     //发生异常时进入此方法    public override void OnException(MethodExecutionArgs args)    {        base.OnException(args);    } //执行方法前执行此方法    public override void OnEntry(MethodExecutionArgs args)    {        base.OnEntry(args);    } //执行方法后执行此方法    public override void OnExit(MethodExecutionArgs args)    {        base.OnExit(args);    }}

注意这里的TestAop这个类必须要是可序列化的,所以要加上[Serializable]特性

(4)在需要拦截功能的地方使用。

在类上面加特性拦截,此类下面的所有的方法都会具有拦截功能。

[TestAop]public class Impc_TM_PLANT : Ifc_TM_PLANT  {      ///       /// 获取或设置服务接口。      ///       private Ic_TM_PLANTService service { get; set; }       public IList Find()      {          DTO_TM_PLANT otest = null;          otest.NAME_C = "test";            //异常,会进入OnException方法      return service.FindAll();      }  }

方法上面加特性拦截,只会拦截此方法。

[TestAop]public IList Find(){    DTO_TM_PLANT otest = null;    otest.NAME_C = "test";    return service.FindAll();}

有没有感觉很简单,很强大,其实这一简单应用,解决我们常见的日志、异常、权限验证等功能简直太小菜一碟了。当然Postsharp可能还有许多更加高级的功能,有兴趣可以深究下。

4、MVC里面的Filter

public class AOPFilterAttribute : ActionFilterAttribute, IExceptionFilter {      public void OnException(ExceptionContext filterContext)     {         throw new System.NotImplementedException();     }     public override void OnActionExecuting(ActionExecutingContext filterContext)     {          base.OnActionExecuting(filterContext);     }      public override void OnActionExecuted(ActionExecutedContext filterContext)     {         base.OnActionExecuted(filterContext);     } }

在controller里面使用该特性:

[AOPFilter]   public JsonResult GetEditModel(string strType)   {       var lstRes = new List>();       var lstResPage = new List();        //.........todo        return Json(new { lstDataAttr = lstRes, PageAttr = lstResPage, lstJsConnections = lstJsPlumbLines }, JsonRequestBehavior.AllowGet);   }

调试可知,在执行GetEditModel(string strType)方法之前,会先执行OnActionExecuting()方法,GetEditModel(string strType)之后,又会执行OnActionExecuted()方法。这在我们MVC里面权限验证、错误页导向、日志记录等常用功能都可以方便解决。

以上就是C#进阶系列——AOP?AOP!的内容,更多相关内容请关注PHP中文网(www.php.cn)!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1432133.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月17日 06:15:21
下一篇 2025年12月17日 06:15:28

相关推荐

  • C#中CLR(公共语言运行时)与IL(中间代码)

    .net平台中的CLR 首先要说明的是,.net平台与c#不是一回事 它是c#,vb.net等程序运行的平台。 CLR是公共语言运行时,是 .NET Framework的重要组成部分。它提供了内存管理、线程管理和异常处理等服务,而且还负责对代码实施严格的类型安全检查,保证了代码的正确性。 事实上,类…

    2025年12月17日
    000
  • C#基础系列:Linq to Xml读写xml

    前言:xml的操作方式有多种,但要论使用频繁程度,博主用得最多的还是linq to xml的方式,觉得它使用起来很方便,就用那么几个方法就能完成简单xml的读写。之前做的一个项目有一个很变态的需求:c#项目调用不知道是什么语言写的一个webservice,然后添加服务引用总是失败,通过代理的方式动态…

    好文分享 2025年12月17日
    000
  • C#基础之内存分配

    1.创建一个对象 一个对象的创建过程主要分为内存分配和初始化两个环节。在.NET中CLR管理的内存区域主要有三部分:栈、GC堆、LOH堆,栈主要用来分配值类型数据。它的管理是有系统控制的,而不是像GC堆那样是由GC控制的。当线程执行完值类型实例所在方法后,这块空间将会被自动释放,一般栈的执行效率高不…

    2025年12月17日
    000
  • C#正则表达式开源工具

    先交代一下背景,最近工作中经常用到正则表达式,而正则表达式这个东西我个人觉得很鸡肋,不用吧,有些功能实现起来会很麻烦。用吧,又不是说工作中经常用到,只是有时候有些需要求用到而已。但是正则表达式只要一段时间不用,就会被遗忘,甚至是忘的一干二净。为了一定程度上解决这个鸡肋的问题,就有了这篇博客和我打算写…

    好文分享 2025年12月17日
    000
  • C# Reflection 反射

    在没使用反射之前,跨项目级的调用普遍的做法是项目级添加引用。 举例:Client 类调用 MysqlHelper 类的话 首先生成 MysqlHelper 项目, 然后在 Client 类中添加 MysqlHelper.dll, 接着在 Client 的方法中实例化,然后调用方法。 使用反射后,可以…

    2025年12月17日
    000
  • C#拾遗之SmtpClient类

    smtpclient类 允许应用程序使用简单邮件传输协议 (SMTP) 发送电子邮件。 命名空间:system.net.mail 属性 ClientCertificates:指定应使用哪个证书来建立安全套接字层(SSL)连接 Credentials:获取或设置用来对发件人进行身份验证的凭证 Deli…

    好文分享 2025年12月17日
    000
  • C# 运算符重载

    您可以重定义或重载 c# 中内置的运算符。因此,程序员也可以使用用户自定义类型的运算符。重载运算符是具有特殊名称的函数,是通过关键字 operator 后跟运算符的符号来定义的。与其他函数一样,重载运算符有返回类型和参数列表。 例如,请看下面的函数: public static Box operat…

    好文分享 2025年12月17日
    000
  • C#接口(Interface)

    C# 接口(Interface) 接口定义了所有类继承接口时应遵循的语法合同。接口定义了语法合同 “是什么” 部分,派生类定义了语法合同 “怎么做” 部分。 接口定义了属性、方法和事件,这些都是接口的成员。接口只包含了成员的声明。成员的定义是派生类的责…

    好文分享 2025年12月17日
    000
  • C#开始使用 LINQ (上)

    LINQ 简介 语言集成查询 (linq) 是 visual studio 2008 和 .net framework 3.5 版中引入的一项创新功能。 传统上,针对数据的查询都是以简单的字符串表示,而没有编译时类型检查或 IntelliSense 支持。此外,您还必须针对以下各种数据源学习一种不同…

    2025年12月17日
    000
  • C#编程之Excel导入、导出(源码下载) (上)

    本篇主要介绍c#的excel导入、导出。 1. 介绍 1.1 第三方类库:NPOI 说明:NPOI是POI项目的.NET 版本,可用于Excel、Word的读写操作。 优点:不用装Office环境。 下载地址:http://npoi.codeplex.com/releases 1.2 Excel结构…

    2025年12月17日
    000
  • C#并发编程·经典实例读书笔记

    前言 最近在看《C# 并发编程 · 经典实例》这本书,这不是一本理论书,反而这是一本主要讲述怎么样更好的使用好目前 C#.NET 为我们提供的这些 API 的一本书,书中绝大部分是一些实例,在日常开发中还是经常会使用到。 书中一些观点还是比较赞同,比如作者说目前绝大多数的图书对关于并发多线程等这些内…

    2025年12月17日
    000
  • C#汉字转拼音(支持多音字)

    之前由于项目需要,中间需要一个汉字转拼音和首拼的功能来做查询,感觉这种功能基本已经成熟化了,于是查找了相关的代码,首先引入眼帘的是下面两篇文章 C# 汉字转拼音(支持GB2312字符集中所有汉字)(http://www.cnblogs.com/cxd4321/p/4203383.html) 【干货】…

    2025年12月17日
    000
  • C#编程基础之序列化

    一、序列化的含义 序列化是将对象处理为字节流以存储对象或传输到内存、数据库或文件。其主要目的是保存对象的状态,以便可以在需要时重新创建对象。相反的过程称为反序列化。 1.1 序列化的工作方式 此图显示序列化的整个过程。 对象被序列化为流。流传递的不仅是数据,还包括有关对象类型的信息,如对象的版本、区…

    2025年12月17日 好文分享
    000
  • C#编程基础之泛型方法解析(上)

    c#2.0引入了泛型这个特性,由于泛型的引入,在一定程度上极大的增强了c#的生命力,可以完成c#1.0时需要编写复杂代码才可以完成的一些功能。但是作为开发者,对于泛型可谓是又爱又恨,爱的是其强大的功能,以及该特性带来的效率的提升,恨的是泛型在复杂的时候,会呈现相当复杂的语法结构。 这种复杂不仅是对于…

    好文分享 2025年12月17日
    000
  • C#编程之Excel导入、导出(源码下载) (下)

    3. Excel导出 3.1 导出流程 3.2 NPOI操作代码 说明:把List转换为Excel 步骤: ①创建一个工作簿(Workbook); ②在工作簿上创建一个工作表(Sheet); ③在工作表上创建第一行(row),第一行为列头,依次写入cellHeard的值(做为列名)。 ④循环遍历Li…

    2025年12月17日
    000
  • C# 程序中嵌入百度地图

    本例是对winform中使用百度地图的简要介绍。百度地图目前支持android开发,ios开发,web开发,服务接口,具体可以参照’百度地图开放平台’。 【动态加载百度地图】涉及到的知识点: WebBrowser控件,此控件是VS自带的控件,使用户可以在WinForm窗体中导…

    2025年12月17日
    000
  • C#拾遗之小知识(一)

    小知识(一) var: 初始化必须有赋值,var i;(错的)。 var类型的对象运行过程中不允许再更改类型。(如:var i=1; i=”hello!” 会出错) var只声明局部变量。 dynamic: 初始化时可以不赋值。 dynamic类型的对象运行过程中可以给该类型…

    好文分享 2025年12月17日
    000
  • C#拾遗之小知识(二)

    小知识(二) 位运算符:     与:&        二进制位均为1时,结果为1,否则0。    二元运算,按位比较。     或:|        二进制位均为0时,结果为0,否则1。    二元运算,按位比较。     异或:^    二进制位相同时,结果为0,否则1。    二元运算…

    好文分享 2025年12月17日
    000
  • C++ 多线程框架(3):消息队列

    之前,多线程一些基本的东西,包括线程创建,互斥锁,信号量,我们都已经封装,下面来看看消息队列 我们尽量少用系统自带的消息队列(比如Linux的sys/msgqueue),那样移植性不是很强,我们希望的消息队列,在消息打包和提取都是用的标准的C++数据结构,当然,你也可以用链表或者是FIFO,那样得先…

    好文分享 2025年12月17日
    000
  • C++ 多线程框架 (2):Mutex 互斥和 Sem 信号量

    互斥和信号量是多线程编程的两个基础,其原理就不详细说了,大家去看看操作系统的书或者网上查查吧。 对于互斥的实现,无论什么操作系统都离不开三个步骤 初始化互斥锁 锁操作 解锁操作 立即学习“C++免费学习笔记(深入)”; 对于不同的系统只是实现的函数有一些不同而已,但是功能其实都大同小异,在锁操作和解…

    好文分享 2025年12月17日
    000

发表回复

登录后才能评论
关注微信