# .NET切面编程——PostSharp

目录

  • 概念
  • 实现方式
  • .Net平台的切面实现
  • PostSharp示例

概念

  • Aspect-Oriented Programming(AOP):想想OOP是不是有些熟悉,AOP翻译过来的意思就是面向切面编程。先来关注一下涉及到的以下几个概念点。
  • 横切关注点:
    存在于项目的绝大多数业务中可以通用的一些辅助性的功能。例如日志、安全、持久化等模块。它们存在于核心业务代码块的各个地方,却又独立于这些核心业务逻辑。
    如图所示:
    Paste_Image.png
  • 切面:
    这些横切关注点的统一抽象。
  • ** 所以面向切面编程,就是将项目的辅助性功能(如日志、异常处理、缓存处理等)与业务逻辑进行分离,把繁琐的辅助性代码抽离出来不用重复Copy,使得程序具备更高的模块化。**

实现方式

  • 静态织入
    即编译时织入,实现原理是对编译器做扩展,使得在代码编译时编译器将切面代码织入到指定的切点。
  • 动态织入
    即运行时织入,编译器在编译时对切面代码和业务代码分别独立编译,而在运行的时候由CLR进行代码混合。

.Net平台的切面实现——PostSharp

  • 为什么选用PostSharp
    1. 轻量级的静态织入实现(可以通过反编译清晰的知道你的代码构成)
    • 使用简单,独立编写切面类,更好的实现模块化,继承自PostSharp提供的各种切面类型的抽象类,并重写其中的拦截方法即可,可以像使用类库内置的Attribute那样使用AOP
    • 对调用方法有更多的控制点,比如输入参数、返回结果、异常捕获等
    • 但是,但是自从2.0版本之后就不免费了
  • PostSharp的切面类
    1. OnMethodBoundaryAspect,针对方法内的各种可能存在切点的情况进行代码注入,实现切面思想,提供了OnEntry,OnExit,OnSuccess,OnException等可重写的虚方法,顾名思义分别是在进入方法、退出方法、方法体成功执行、方法内发生异常的拦截。后续的实例是通过OnEntry来修改方法的输入参数来展示的。
    • OnFiledAccessAspect,对Filed的读写进行拦截处理,提供了OnGetValue,OnSetValue的虚方法。
    • OnExceptionAspect,实现异常的捕获。
    • OnMethodInvocationAspect,方法调用拦截,提供OnInvocation虚方法。
    • ImplementMethodAspect,用于extern方法、abstract类的方法进行拦截。
  • PostSharp的版本差异
    3.0版本是个分水岭,目前最新版本是5.0.28,他们的使用方式有一小点差异。
    • 3.0版本之前下载了安装包,在项目中引用postsharp.dll,之后编码就可以了
    • 但是3.0之后是类似于VS插件的方式工作的,下载postsharp安装包安装之后会在vs的菜单中新增一个postsharp菜单(如下图),可以进行一些设置,在使用的时候不再是引用,而是需要项目上右击“添加postsharp到项目”

<img src="http://upload-images.jianshu.io/upload_images/3566275-b89bf795f818e740.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" />


PostSharp示例

  1. 做前期准备工作,从PostSharp官网下载最新版本的安装包,并安装。
  • 打开VisualStudio,新建解决方案,添加命令行项目。项目上右键后点击“Add PostSharp to project”后在弹出窗口按提示操作,会发现已经多了postsharp的引用。

<img src="http://upload-images.jianshu.io/upload_images/3566275-33826a58ba5345fd.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" style="width:45%"/><img src="http://upload-images.jianshu.io/upload_images/3566275-776a74ab91f1303c.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" style="width:45%"/>

  • 添加AspectAttribute切面类,实现对核心方法的输入参数修改。需要注意的是postsharp提供的几种切面类型都是继承自Attribute基类,而且是通过对要实现拦截的类或方法添加特性的方式实现切面思想的。所以我们的切面类 需要按照约定以xxxAttribute的格式命名

      [Serializable]
      [AttributeUsage(AttributeTargets.Method, AllowMultiple = true, Inherited = true)]
      public class AspectAttribute : OnMethodBoundaryAspect
      {
          //方法进入时
          public override void OnEntry(MethodExecutionArgs args)
          {
              //修改输入参数
              args.Arguments[0] = "jingdong";
              //设置方法是否继续执行或退出,若设置的是FlowBehavior.Return方法会直接退出,不执行后续的所有代码。
              args.FlowBehavior = FlowBehavior.Continue;
          }
    
          //方法离开时
          public override void OnExit(MethodExecutionArgs args)
          {
              Console.WriteLine("exit");
          }
    
          //方法成功执行时
          public override void OnSuccess(MethodExecutionArgs args)
          {
              Console.WriteLine("success");
          }
      }
    
  • 在program中添加我们的核心方法Start,打印输入参数。在需要实现拦截的方法添加上一个步骤中实现的切面类特性

      public class Program
      {
          static void Main(string[] args)
          {
              var arg = "tmall";
              Console.WriteLine($"original argument:{arg}");
              Start(ref arg);
              Console.Read();
          }
    
          [Aspect]
          static void Start(ref string arg)
          {
              Console.WriteLine($"real argument:{arg}");
              Thread.Sleep(1000);
              Console.WriteLine("finished");
          }
      }
    
  • 由此就简单通过postsharp实现了对方法的拦截,修改输入参数,监听方法成功执行以及退出。看下执行结果,成功的篡改了输入参数,并且可以看出拦截方法的方法体是先于OnSuccess执行的,OnSuccess的拦截执行完成之后才是OnExit。

  • 之前有说过postsharp是静态织入来实现AOP编程的,那么肯定是通过编译器在编译的时候对代码进行了织入,可以通过反编译exe文件来看下。


可以看到start方法被加入了若干行代码。

  • 有一个小点需要注意,使用postsharp的时候对切面类必须添加Serializable特性,否则在编译的时候就会报错。

<img src="http://upload-images.jianshu.io/upload_images/3566275-b5cdab3280ceb4a9.png?imageMogr2/auto-orient/strip%7CimageView2/2/w/1240" width="70%"/>

  • 小结

文章内的小例子主要是为了说明postsharp实现AOP的基本原理,以及实现过程。并没有如同实际项目中使用AOP来做一些日志、安全、持久化之类的辅助功能。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 200,176评论 5 469
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 84,190评论 2 377
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 147,232评论 0 332
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 53,953评论 1 272
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 62,879评论 5 360
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,177评论 1 277
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,626评论 3 390
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,295评论 0 254
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,436评论 1 294
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,365评论 2 317
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,414评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,096评论 3 315
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,685评论 3 303
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,771评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,987评论 1 255
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,438评论 2 346
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,032评论 2 341

推荐阅读更多精彩内容