.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技 ...
随机推荐
- RabbitMQ 安装 rabbitmq_delayed_message_exchange插件
rabbitmq_delayed_message_exchange插件主要是实现延迟队列 一.下载插件 下载地址:http://www.rabbitmq.com/community-plugins.h ...
- MySQL与DevC++的连接问题
0.引言 MySQL作为当前的一个主流的开源的关系型数据库,受到大家的广泛关注.DevC++也作为一个开源的简单的C++编辑器,有着比VS更简便更快捷的编译.但是目前网上,却很少有对 MySQL连接D ...
- 关于Myeclipse的MyEclipse:Java was started but returned exit code=-1 错误
我们在安装MyEclipse后有时会遇到这样一个问题,可以进入主界面软件也属于激活状态,但是过一会会报错, 并弹出MyEclipse:Java was started but returned exi ...
- 15-RUN vs CMD vs ENTRYPOINT
RUN.CMD 和 ENTRYPOINT 这三个 Dockerfile 指令看上去很类似,很容易混淆.本节将通过实践详细讨论它们的区别. 简单的说: RUN 执行命令并创建新的镜像层,RUN 经常用于 ...
- 五、Django之视图和模板-Part 3
一.概述 一个视图就是一个页面,通常提供特定的功能,使用特定的模版.列如:在一个博客应用中,你可能会看到下列视图: 博客主页:显示最新发布的一些内容 每篇博客的详细页面:博客的永久链接 基于年的博客页 ...
- python 思维导图-linux命令
linux基础 linux关于文件/目录的终端命令 linux关于文件内容处理的终端命令
- Ubuntu解压zip包中文乱码
解决方法:通过unar 工具解压 步骤一: 安装unar: sudo apt-get install unrar 步骤二: 解压(以test.zip为例):unar test.zip 解压成功,乱码问 ...
- mysql 从 frm 文件恢复 table 表结构的3种方法
mysql 正常运行的时候,查看 table 的结构并不是困难的事. 但是有时 mysql 发生故障,这种方法便不再可行. 当遇到故障,通常使用新的 mysql 实例来恢复当前的数据. 建表是非常重要 ...
- pytorch 如何使用tensorboard实时查看曲线---- tensorboardX简介
用惯了tensorflow的小伙伴肯定都用过tensorboard工具吧.虽然Facebook也推出了visdom,但是在一次不小心误触clear之后,我放弃了这个工具(页面的一个clear按钮我本来 ...
- AS的使用技巧
title: AS的使用技巧 date: 2016-04-01 23:34:11 tags: [AndroidStudio] categories: [Tool,IDE] --- 概述 本文记录如何使 ...





