本文摘自 博文--《Spring设计思想》AOP设计基本原理
0、前言
Spring 提供了AOP(Aspect Oriented Programming) 的支持, 那么,什么是AOP呢?本文将通过一个另外一个角度来诠释AOP的概念,帮助你更好地理解和使用Spring AOP。
读完本文,你将了解到:
1. Java程序运行在JVM中的特征
当我们在某个类Foo中写好了一个main()方法,然后执行java Foo,你的Java程序之旅就开启了,如下:
在某个特定的时间点,一个Main线程内的栈会呈现如下图所示的情况:
从线程栈的角度来看,我们可以看到,JVM处理Java程序的基本单位是方法调用
。实际上,JVM执行的最基本单位的指令(即原子操作)是汇编语言性质的机器字节码。这里之所以讲方法调用
时Java程序的基本执行单位,是从更宏观的角度看待的。
如何获取到虚拟机线程栈中的内容(即方法调用过程)?
试想一下,如何能够获取到JVM线程栈中的方法调用的内容? 我相信所有的Java programmer都知道这个答案。Java Programmer几乎每天都能看到它------当我们的代码抛出异常而未捕获或者运行时出现了Error错误时,我们会受到一个非常讨厌的Log信息,如下:
当然,除了代码抛出异常外,我们还是可以其他方式察觉JVM线程栈内的内容。可以通过Thread.dumpStack()方法创建一个假的Exception实例,然后将这个Exception实例记录的当前线程栈的内容输出到标准错误流中。例如我在某处代码里执行了Thread.dumpStack()方法,输出了如下的结果:
2. Java程序执行流 【了解AOP、连接点(Join Point)、切入点(point cut) 的概念 】
如果从虚拟机线程栈的角度考虑Java程序执行的话,那么,你会发现,真个程序运行的过程就是方法调用的过程。我们按照方法执行的顺序,将方法调用排成一串,这样就构成了Java程序流。
3. 引入了代理模式的Java程序执行流(AOP实现的机制)
4. Spring AOP的工作原理
AOP对这个方法调用的编程,就是针对这三个阶段插入自己的业务代码。
现在我们假设当前RealSubject这个角色的类是 org.luanlouis.springlearning.aop.FooService ,当前这个连接点对应的方法签名是:public void foo()。那么上述的代理对象的三个阶段将会有以下的处理逻辑:
1. 在调用真正对象的方法之前,
proxy会告诉Spring AOP: "我将要调用类org.luanlouis.springlearning.aop.FooService 的public void foo() ,在调用之前,你有什么处理建议吗?";
Spring AOP这时根据proxy提供的类名和方法签名,然后拿这些信息尝试匹配是否在其感兴趣的切入点内,如果在感兴趣的切入点内,Spring AOP会返回 MethodBeforeAdvice处理建议,告诉proxy应该执行的操作;
2. 在调用真正对象的方法过程中,如果抛出了异常,需要做什么?
proxy告诉Spring AOP: “我调用类org.luanlouis.springlearning.aop.FooService 的public void foo()过程中抛出了异常,你有什么处理建议?”
Spring AOP根据proxy提供的类型和方法签名,确定了在其感兴趣的切入点内,则返回相应的处理建议ThrowsAdvice,告诉proxy这个时期应该采取的操作。
3.在调用真正对象的方法后,返回了结果了,需要做什么?
proxy告诉Spring AOP:"我调用类org.luanlouis.springlearning.aop.FooService 的public void foo()结束了,并返回了结果你现在有什么处理建议?";
Spring AOP 根据proxy提供的类型名和方法签名,确定了在其感兴趣的切入点内,则返回AfterReturingAdivce处理建议,proxy得到这个处理建议,然后执行建议;
上述的示意图中已经明确表明了Spring AOP应该做什么样的工作:根据proxy提供的特定类的特定方法执行的特定时期阶段给出相应的处理建议。要完成该工作,Spring AOP应该实现: