AOP 和 OOP,在我看来是两种相辅相成的技术,作为OOP的补充,AOP 有着自己特殊的应用场景。

假设,我们需要在Service层实现以下几项基本功能:

/// <para>1、自动管理数据库连接[可选]</para>
/// <para>2、自动管理数据库事务,当接收到异常后(无论什么异常)事务将自动回滚[可选]</para>
/// <para>3、服务级加锁[必选]</para>
/// <para>4、以统一方式处理 服务异常 及 错误, 包括数据库异常 和 主动抛出的异常[必选]</para>

解释:
1、在 执行Service方法前 打开数据库连接, 在 执行Service方法后 关闭数据库连接
2、在 执行Service方法前 Begin数据库事务, 在 执行Service方法后 Commit数据库事务, Catch异常后 RollBack数据库事务

3、将 整个Service方法 lock 进去,lock Service 的私有静态对象,以达到服务级方法的 线程安全及同步工作
4、捕获Service方法中所有未捕获的异常,捕获异常后,如果需要将自动关闭连接和回滚事务。并记录异常信息。
即、主动报告错误时,只需要抛出异常即可。

为了 实现如上的功能,并能简单方便实现,而且不打破现有的C#编码规范。
所以,引入AOP、 使用 Attribute 为方法 指定增强对象,
以便在调用Service方法前,执行方法的前置增强(包括打开连接、开启事务等)
在调用Service方法后,执行方法的后置增强(包括关闭连接、提交事务等)
及 对整个调用方法实现 Try...Catch异常捕获 和 Lock 加锁。

C# 引入了 Proxy (代理)的概念,即 System.Runtime.Remoting.Proxies.RealProxy 提供了 代理的基本功能。利用该对象可以自己实现AOP编程。

RealProxy 可以可以为任何 “直接或间接继承于 System.MarshalByRefObject” 的类型 提供代理。
RealProxy 可以为指定类型创建一个代理对象, 被创建的代理对象的类型 可以看做是 指定类型的 子类(但 被指定的类型可以是密封类)。
【PS: 看做子类,更容易理解,本质上为被创建的代理对象的类型 和 指定类型直接为 组合关系,并不是继承关系 】

RealProxy 的工作原理:
假设:
T 为 需要被代理的类型, t 为对象
ProxyT 为 被创建的代理类型, proxyT 为对象

T 类型中存在 成员方法 Test();
ProxyT 继承于 T【实际上不为继承关系,应该为组合,为方便理解看做继承关系】, ProxyT 同样也存在方法 Test

当执行如下代码时:
proxyT.Test();

.NET runtime 会自动调用 System.Runtime.Remoting.Proxies.RealProxy.Invoke(...)方法。
而该方法为抽象方法,自己重写该方法,在方法内部调用 t.Test()。
在调用之前、执行前置增强;在调用之后、执行后置增强; 及 其他处理操作。
由此可实现 AOP 编程,织入增强。

自定义的RealProxy

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Runtime.Remoting.Messaging;
using System.Runtime.Remoting.Proxies; namespace AOPDemo.Common
{
public class DelayProxy<T> : RealProxy
{
private static object objLock = new object(); /// <summary>
/// 被代理的对象
/// </summary>
private T target; public DelayProxy(T target)
: base(typeof(T))
{
this.target = target; } public override IMessage Invoke(IMessage msg)
{
IMethodCallMessage callMessage = (IMethodCallMessage)msg; Console.WriteLine("方法被调用前"); Console.WriteLine("调用方法名:" + callMessage.MethodName); IMessage message = DelayProxyUtil.InvokeBeProxy(this.target, callMessage); Console.WriteLine("方法被调用后"); return message;
} } }

RealProxy

辅助工具类

using System;
using System.Collections.Generic;
using System.Linq;
using System.Web;
using System.Runtime.Remoting.Proxies;
using System.Runtime.Remoting.Messaging;
using System.Reflection; namespace AOPDemo.Common
{ /// <summary>
/// 延迟初始化代理工具类
/// </summary>
public static class DelayProxyUtil
{
/// <summary>
/// 调用被代理对象中方法,返回 被代理对象的 方法返回值
/// <para>支持 out ref 参数</para>
/// </summary>
/// <param name="target"></param>
/// <param name="callMessage"></param>
/// <returns></returns>
public static IMessage InvokeBeProxy(object target, IMethodCallMessage callMessage)
{
var args = callMessage.Args; object returnValue = callMessage.MethodBase.Invoke(target, args); return new ReturnMessage(returnValue, args, args.Length, callMessage.LogicalCallContext, callMessage);
} /// <summary>
/// 向上层抛出异常
/// </summary>
/// <param name="ex"></param>
/// <param name="callMessage"></param>
/// <returns></returns>
public static IMessage ReturnExecption(Exception ex, IMethodCallMessage callMessage)
{
return new ReturnMessage(ex, callMessage);
} /// <summary>
/// 获取对象的代理
/// </summary>
/// <param name="type"></param>
/// <param name="instance"></param>
/// <param name="delay"></param>
/// <returns></returns>
public static object GetTransparentProxy(Type type, object instance)
{
Type tmpType = typeof(DelayProxy<>); tmpType = tmpType.MakeGenericType(type); RealProxy proxy = Activator.CreateInstance(tmpType, new object[] { instance }) as RealProxy; return proxy.GetTransparentProxy();
} }
}

辅助工具类

简单的Demo

    public class HomeController : Controller
{
//
// GET: /Home/ public ActionResult Index()
{
Service service = new Service(); Service proxy = Common.DelayProxyUtil.GetTransparentProxy(typeof(Service), service) as Service; proxy.Test(); return View();
} } public class Service : MarshalByRefObject
{
public void Test()
{
Console.WriteLine("调用Test方法");
}
}

由于例子很简单,就不上传源码了。

未完待续...

自己实现简单的AOP(一)简介的更多相关文章

  1. 自己实现简单的AOP(二)引入Attribute 为方法指定增强对象

    话续前文 : 自己实现简单的AOP(一)简介 在前一篇文章中,对AOP的实现方式做了一个简单介绍.接下来,引入Attribute 为方法指定增强对象,由此实现一个简单的AOP. 注意:指定的是增强对象 ...

  2. 自己实现简单的AOP(四)自动初始化代理对象

    前面三篇随笔,已经完成了AOP的核心功能,但 代理对象的初始化还是有些麻烦,本文将解决该问题. Demo 片段如下: public class HomeController : Controller ...

  3. 从头认识Spring-3.8 简单的AOP日志实现(注解版)-扩展添加检查订单功能,以便记录并检測输入的參数

    这一章节我们讨论一下扩展添加检查订单功能,以便记录并检測输入的參数. 1.domain 蛋糕类: package com.raylee.my_new_spring.my_new_spring.ch03 ...

  4. aop学习总结二------使用cglib动态代理简单实现aop功能

    aop学习总结二------使用cglib动态代理简单实现aop功能 模拟业务需求: 1.拦截所有业务方法 2.判断用户是否有权限,有权限就允许用户执行业务方法,无权限不允许用户执行业务方法 (判断是 ...

  5. 从头认识Spring-3.1 简单的AOP日志实现-某方法之前的前后记录日志

    这一章节我们引入简单的AOP日志实现. 1.domain 蛋糕类: package com.raylee.my_new_spring.my_new_spring.ch03.topic_1_1; pub ...

  6. 从头认识Spring-3.4 简单的AOP日志实现-扩展添加检查订单功能,以便记录并检測输入的參数

    这一章节我们再上一个章节的基础上加上一个检查订单功能 1.domain 蛋糕类: package com.raylee.my_new_spring.my_new_spring.ch03.topic_1 ...

  7. Android埋点方案的简单实现-AOP之AspectJ

    个人博客 http://www.milovetingting.cn Android埋点方案的简单实现-AOP之AspectJ AOP的定义 AOP为Aspect Oriented Programmin ...

  8. 动手造轮子:实现一个简单的 AOP 框架

    动手造轮子:实现一个简单的 AOP 框架 Intro 最近实现了一个 AOP 框架 -- FluentAspects,API 基本稳定了,写篇文章分享一下这个 AOP 框架的设计. 整体设计 概览 I ...

  9. 自己实现简单的AOP(三) 实现增强四项基本功能

    前面的两篇随笔,都是只是个铺垫,真正实现增强四项基本功能的重头戏,在本篇随笔中, 本文将通过AOP实现如下的四个基本功能: /// <para>1.自动管理数据库连接[可选]</pa ...

随机推荐

  1. 使用CSS使内容垂直居中的N中方法。

    使用css+div使页面内容水平居中的方法大家并不陌生,那么如何使内容垂直居中呢? OK,下面进入正题,不如我们使用做高中数学题时经常用的思想:分情况讨论.   1.当待垂直居中的DIV高宽为已知时: ...

  2. Android什么时候进行View中Background的加载

    对大多数Android的开发者来说,最经常的操作莫过于对界面进行布局,View中背景图片的加载是最经常做的.但是我们很少关注这个过程,这篇文章主要解析view中背景图片加载的流程.了解view中背景图 ...

  3. .Net组件程序设计之异步调用

    .Net组件程序设计之异步调用 说到异步调用,在脑海中首先想到就是BeginInvoke(),在一些常用对象中我们也会常常见到Invoke()和BeginInvoke(), 要想让自己的组件可以被客户 ...

  4. Express4 启航指南

    确实有感而发,Nodejs真的发展太快了,这么说的原因有两点:自己去年冬天买了本<了不起的Node.js>,里面介绍Express的版本还是2.x.x:前些天小伙伴买了本<Node. ...

  5. 微信小程序产品定位及功能介绍

    产品定位及功能介绍 微信小程序是一种全新的连接用户与服务的方式,它可以在微信内被便捷地获取和传播,同时具有出色的使用体验. 小程序注册 注册小程序帐号 在微信公众平台官网首页(mp.weixin.qq ...

  6. C#设计模式-享元模式

    在软件开发过程,如果我们需要重复使用某个对象的时候,如果我们重复地使用new创建这个对象的话,这样我们在内存就需要多次地去申请内存空间了,这样可能会出现内存使用越来越多的情况,这样的问题是非常严重,然 ...

  7. WCF学习之旅—WCF第二个示例(六)

    第五步,创建数据服务 在“解决方案资源管理器”中,使用鼠标左键选中“SCF.WcfService”项目,然后在菜单栏上,依次选择“项目”.“添加新项”. 在“添加新项”对话框中,选择“Web”节点,然 ...

  8. Sublime Text执行js

    Sublime Text执行js 在Build Sytem添加以下内容: { "cmd": ["node", "$file"], " ...

  9. 关于HTML5 Audio线程问题

    移动端果然很坑! 在移动端IOS平台上用new Audio每次调用这个API都会创建一个新的线程,而且还不能销毁...直到拖死应用 后来改进了换了AudioContext,线程问题解决了 https: ...

  10. 深入学习jQuery选择器系列第六篇——过滤选择器之状态选择器

    × 目录 [1]焦点状态 [2]哈希状态 [3]动画状态[4]显隐状态 前面的话 过滤选择器的内容非常多,本文介绍过滤选择器的最后一部分——状态选择器 焦点状态 :focus :focus选择器选择当 ...