# .NET切面编程——PostSharp
目录
- 概念
- 实现方式
- .Net平台的切面实现
- PostSharp示例
概念
- Aspect-Oriented Programming(AOP):想想OOP是不是有些熟悉,AOP翻译过来的意思就是面向切面编程。先来关注一下涉及到的以下几个概念点。
- 横切关注点:
存在于项目的绝大多数业务中可以通用的一些辅助性的功能。例如日志、安全、持久化等模块。它们存在于核心业务代码块的各个地方,却又独立于这些核心业务逻辑。
如图所示:
Paste_Image.png - 切面:
这些横切关注点的统一抽象。 - 所以面向切面编程,就是将项目的辅助性功能(如日志、异常处理、缓存处理等)与业务逻辑进行分离,把繁琐的辅助性代码抽离出来不用重复Copy,使得程序具备更高的模块化。
实现方式
- 静态织入
即编译时织入,实现原理是对编译器做扩展,使得在代码编译时编译器将切面代码织入到指定的切点。 - 动态织入
即运行时织入,编译器在编译时对切面代码和业务代码分别独立编译,而在运行的时候由CLR进行代码混合。
.Net平台的切面实现——PostSharp
- 为什么选用PostSharp
- 轻量级的静态织入实现(可以通过反编译清晰的知道你的代码构成)
- 使用简单,独立编写切面类,更好的实现模块化,继承自PostSharp提供的各种切面类型的抽象类,并重写其中的拦截方法即可,可以像使用类库内置的Attribute那样使用AOP
- 对调用方法有更多的控制点,比如输入参数、返回结果、异常捕获等
- 但是,但是自从2.0版本之后就不免费了
- PostSharp的切面类
- 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到项目”

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


添加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特性,否则在编译的时候就会报错。

小结
文章内的小例子主要是为了说明postsharp实现AOP的基本原理,以及实现过程。并没有如同实际项目中使用AOP来做一些日志、安全、持久化之类的辅助功能。
# .NET切面编程——PostSharp的更多相关文章
- .NET 切面编程 PostSharp
目录 概念 实现方式 .Net平台的切面实现 PostSharp示例 概念 Aspect-Oriented Programming(AOP):想想OOP是不是有些熟悉,AOP翻译过来的意思就是面向切面 ...
- 从壹开始前后端分离【 .NET Core2.0 +Vue2.0 】框架之十 || AOP面向切面编程浅解析:简单日志记录 + 服务切面缓存
代码已上传Github+Gitee,文末有地址 上回<从壹开始前后端分离[ .NET Core2.0 Api + Vue 2.0 + AOP + 分布式]框架之九 || 依赖注入IoC学习 + ...
- AOP面向切面编程C#实例
原创: eleven 原文:https://mp.weixin.qq.com/s/8klfhCkagOxlF1R0qfZsgg [前言] AOP(Aspect-Oriented Programming ...
- C# 中使用面向切面编程(AOP)中实践代码整洁
1. 前言 最近在看<架构整洁之道>一书,书中反复提到了面向对象编程的 SOLID 原则(在作者的前一本书<代码整洁之道>也是被大力阐释),而面向切面编程(Aop)作为面向对象 ...
- 【ABP杂烩】面向切面编程(AOP)知识总结
目录 1.存在问题 2.AOP的概念 3.AOP应用范围 3.AOP实现方式 4.应用举例 5.结束语 本文在学习[老张的哲学]系列文章AOP相关章节后,自己归纳总结的笔记. 1.存在问题 最近,其它 ...
- C# 中使用面向切面编程(AOP)中实践代码整洁(转)
出处:https://www.cnblogs.com/chenug/p/9848852.html 1. 前言 最近在看<架构整洁之道>一书,书中反复提到了面向对象编程的 SOLID 原则( ...
- Java 面向切面编程(Aspect Oriented Programming,AOP)
本文内容 实例 引入 原始方法 装饰者模式 JDK 动态代理和 cglib 代理 直接使用 AOP 框架--AspectWerkz 最近跳槽了,新公司使用了 AOP 相关的技术,于是查点资料,复习一下 ...
- AOP(面向切面编程概念,本文为翻译)
AOP是什么 AOP为Aspect Oriented Programming的缩写.AOP是OOP的延续,是软件开发中的一个热点,也是Spring框架中的一个重要内容,是函数式编程的一种衍生范型.利用 ...
- C# 面向切面编程 AOP
AOP(Aspect Oriented Programming) 面向切面编程 起源 引言 http://wayfarer.cnblogs.com/articles/241012.html AOP技 ...
随机推荐
- 从RGB色转为灰度色算法
一.基础 对于彩色转灰度,有一个很著名的心理学公式: Gray = R*0.299 + G*0.587 + B*0.114 二.整数算法 而实际应用时,希望避免低速的浮点运算,所以需要整数算法. 注 ...
- HTML5 Web Worker简单使用
Web Workers 是 HTML5 提供的一个javascript多线程解决方案,我们可以将一些大计算量的代码交由web Worker运行而不冻结用户界面. 一:如何使用Worker Web Wo ...
- vue-schart : vue.js 的图表组件
介绍 vue-schart 是使用vue.js封装了sChart.js图表库的一个小组件.支持vue.js 1.x & 2.x 仓库地址:https://github.com/lin-xin/ ...
- sleep()方法和wait()方法之间有什么差异?
sleep()方法用被用于让程序暂停指定的时间,而wait()方法被调用后,线程不会自动苏醒,需要别的线程调用同一个对象上的notify()或者nofifyAl()方法 主要的区别是,wait()释放 ...
- 聊聊RocksDB Compact
| 导语 对于 LevelCompact 策略,RocksDB会根据每一层不同的策略计算出CompactScore,根据CompactScore大小来决定那一层将会优先进行Compact,然后选择Le ...
- Linux 特殊用户权限 suid,sgid, sticky
每个进程会维护有如下6个ID: 真实身份 : real UID, readl GID --> 登录 shell 使用的身份 有效身份 : effective UID, effective GID ...
- document事件及例子
一.关于鼠标事件:onclick:鼠标单击触发 ondbclick:鼠标双击触发 onmouseover:鼠标移上触发 onmouseout:鼠标离开触发 onmousemove:鼠标移动触发 二.关 ...
- httpClient 发送请求后解析流重用的问题(HttpEntity的重用:BufferedHttpEntity)
使用场景: 项目中使用httpClient发送一次http请求,以流的方式处理返回结果,开始发现返回的流只能使用一次,再次使用就会出错,后来看了一些解决方案,EntityUtils.consume(r ...
- Swift字符串插值
字符串插值是一种全新的构建字符串的方式,可以在其中包含常量.变量.字面量和表达式.您插入的字符串字面量的每一项都被包裹在以反斜线为前缀的圆括号中: let multiplier = let messa ...
- C++基础之引用与指针的区别与联系、常引用使用时应注意的问题
什么是引用? 引用就是对变量起一个别名,而变量还是原来的变量,并没有重新定义一个变量.例如下面的例子: #include<iostream> using namespace std; ...





