AspectJ本质剖析
AOP一共有两种方式,spring默认使用的是动态代理(JDK自带的动态代理或者使用cglib的动态代理)和静态代理(ASPECTJ)
http://blog.csdn.net/zhao9tian/article/details/37762389
http://www.ibm.com/developerworks/cn/java/l-aspectJ/index.html中介绍了What is AspectJ 。
- AspectJ是一个代码生成工具(Code Generator)。
- AspectJ语法就是用来定义代码生成规则的语法。您如果使用过Java Compiler Compiler (JavaCC),您会发现,两者的代码生成规则的理念惊人相似。
- AspectJ有自己的语法编译工具,编译的结果是Java Class文件,运行的时候,classpath需要包含AspectJ的一个jar文件(Runtime lib)。
- ....
看了上面几点,我就想看看它怎么把代码生成了。现在做一个试验。
一个类(包括main函数):Speaker.java
- package test.aspectj;
- public class Speaker
- {
- public void speak()
- {
- System.out.println("[Speaker] bla bla ");
- }
- public static void main(String[] args)
- {
- Speaker speaker = new Speaker();
- speaker.speak();
- }
- }
一个aspect:AspectObserver.java
- package test.aspectj;
- public aspect AspectObserver
- {
- pointcut speakerSpeak():
- call(void *Speaker.speak());
- before() : speakerSpeak() {
- System.out.println("[AspectObserver] speaker is about to speak!");
- }
- after() returning() : speakerSpeak() {
- System.out.println("[AspectObserver] speaker has completed his speech!");
- }
- }
以上都是源码部分哦。
运行结果:
- [AspectObserver] speaker is about to speak!
- [Speaker] bla bla
- [AspectObserver] speaker has completed his speech!
说明程序是正常运作的哦。
好了,下面,做三个操作:
1、将以上的编译成的class文件打包成apectjtest.jar文件。
说明:可以使用ajdt的eclipse插件带的导出功能,Export --> JAR file with ApectJ support
2、新建一个AspectJ工程,将apectjtest.jar加入类路径,使用jad来反编译Speaker.class和AspectObserver.class
得到反编译后的源码:
Speaker.class
- /*jadclipse*/// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
- // Jad home page: http://www.kpdus.com/jad.html
- // Decompiler options: packimports(3) radix(10) lradix(10)
- // Source File Name: Speaker.java
- package test.aspectj;
- import java.io.PrintStream;
- // Referenced classes of package test.aspectj:
- // AspectObserver
- public class Speaker
- {
- public Speaker()
- {
- }
- public void speak()
- {
- System.out.println("[Speaker] bla bla ");
- }
- public static void main(String args[])
- {
- Speaker speaker = new Speaker();
- AspectObserver.aspectOf().ajc$before$test_aspectj_AspectObserver$1$b2b6354();
- speaker.speak();
- AspectObserver.aspectOf().ajc$afterReturning$test_aspectj_AspectObserver$2$b2b6354();
- }
- }
AspectObserver.class
- /*jadclipse*/// Decompiled by Jad v1.5.8g. Copyright 2001 Pavel Kouznetsov.
- // Jad home page: http://www.kpdus.com/jad.html
- // Decompiler options: packimports(3) radix(10) lradix(10)
- // Source File Name: AspectObserver.aj
- package test.aspectj;
- import java.io.PrintStream;
- import org.aspectj.lang.NoAspectBoundException;
- public class AspectObserver
- {
- public AspectObserver()
- {
- }
- void ajc$pointcut$$speakerSpeak$44()
- {
- }
- public void ajc$before$test_aspectj_AspectObserver$1$b2b6354()
- {
- System.out.println("[AspectObserver] speaker is about to speak!");
- }
- public void ajc$afterReturning$test_aspectj_AspectObserver$2$b2b6354()
- {
- System.out.println("[AspectObserver] speaker has completed his speech!");
- }
- public static AspectObserver aspectOf()
- {
- if(ajc$perSingletonInstance == null)
- throw new NoAspectBoundException("test_aspectj_AspectObserver", ajc$initFailureCause);
- else
- return ajc$perSingletonInstance;
- }
- public static boolean hasAspect()
- {
- return ajc$perSingletonInstance != null;
- }
- private static void ajc$postClinit()
- {
- ajc$perSingletonInstance = new AspectObserver();
- }
- private static Throwable ajc$initFailureCause;
- public static final AspectObserver ajc$perSingletonInstance;
- static
- {
- try
- {
- ajc$postClinit();
- }
- catch(Throwable throwable)
- {
- ajc$initFailureCause = throwable;
- }
- }
- }
3、运行一下Speaker.class
得到结果:
- [AspectObserver] speaker is about to speak!
- [Speaker] bla bla
- [AspectObserver] speaker has completed his speech!
结果跟源码运行是一样的哦(不一样就是你的人品问题咯!!)
分析一下,先理解AspectJ编译器为我们做了什么事情:
二、AspectJ发现一些文件含有aspect的定义,在这个例子里,就是AspectObserver的定义;这些aspect就是代码生成规则。
三、AspectJ根据这些aspect代码生成规则,修改添加你的源代码。在这个例子里,源码是修改成怎样了?比较一下反编译后的代码和源码便知。
四、AspectJ读取AspectObserver的定义,发现了一个pointcut--speakerSpeak();这个pointcut的定义是call(void *Speaker.speak()),表示所有对Speaker类的speak方法的执行点。
五、AspectJ继续读取AspectObserver的定义,发现了一个before(),这在AspectJ中叫做Advice。Advice允许你在某个类的方法的调用之前或调用之后,加入另外的代码。加入的代码是什么?比较一下反编译后的代码和源码吧。 AspectJ继续读取AspectObserver的定义.
好了,回头看看反编译后的Speaker.class,与源码Speaker.java差别在哪呢?主要在main函数中,方法被调用的前后
- public static void main(String args[])
- {
- Speaker speaker = new Speaker();
- AspectObserver.aspectOf().ajc$before$test_aspectj_AspectObserver$1$b2b6354();//多了这行
- speaker.speak();
- AspectObserver.aspectOf().ajc$afterReturning$test_aspectj_AspectObserver$2$b2b6354();//多了这行
- }
不同的地方就是多出了两行,虽然有$和数字,但是很容易看出,这同我们平常写的java代码差不了多少。
- AspectObserver.aspectOf().ajc$before$test_aspectj_AspectObserver$1$b2b6354();
这行代码中,看起来不就是AspectObserver类调用了静态方法aspectOf()吗,接着aspectOf()的返回对象又调用ajc$before$test_aspectj_AspectObserver$1$b2b6354()方法吗?
回来看看AspectObserver.class的反编译代码,哈哈,这就对了,aspectOf返回的是AspectObserver的一个实例,返回实例在调用实例方法ajc$before$test_aspectj_AspectObserver$1$b2b6354()。ajc$before$test_aspectj_AspectObserver$1$b2b6354()是干什么的呢?呵呵,不正是我们要的拦截方法所要做的操作吗?
- System.out.println("[AspectObserver] speaker is about to speak!");
同样,可知
- AspectObserver.aspectOf().ajc$afterReturning$test_aspectj_AspectObserver$2$b2b6354();
是怎么一回事啦。
这正是应了上面所提的两点:
- AspectJ是一个代码生成工具(Code Generator)。
- AspectJ语法就是用来定义代码生成规则的语法。
用这两点去感受apectj在程序里该如何去运用使用,就清晰多啦!
AspectJ本质剖析的更多相关文章
- java8学习之内部迭代与外部迭代本质剖析及流本源分析
关于Stream在Java8中是占非常主要的地位的,所以这次对它进行进一步探讨[这次基本上都是偏理论的东东,但是理解它很重要~],其实流跟咱们数据库学习当中的sql语句的特点是非常非常之像的,为什么这 ...
- 【转】self.myOutlet=nil、viewDidUnload、dealloc的本质剖析
对于iphone开发人员来说,内存管理是极为重要的技巧,哪怕程序的功能再强大,设计再漂亮,如果内存控制不好,也难逃程序莫名退出的噩运,这与网页开发是完全不同的. 内存控制里面有很多门道,在这里分析一下 ...
- C++中的引用
一,C++中引用的基础知识 1.引用的基本概念 1.所谓的引用其实就是对变量起“别名”.引用和变量对应得是相同的内存,修改引用的值,变量的值也会改变,和指针类似. 2.引用在定义的时候必须要初始化,初 ...
- LINQ的高级应用
---恢复内容开始--- 本文不想罗列linq的通俗使用方法.因为很多博文都已经写得很详细了. 此处直接贴出源码,如果有需要的朋友可以参考,希望更多的朋友能够补充更多的linq的高级应用. 源码如下: ...
- C++ 学习路线和看法
C++ 学习路线和看法 原文地址:http://shundacao.blog.163.com/blog/static/1340404812010101982751101/ C++大体分为C++ ...
- MySQL数据库事务各隔离级别加锁情况--read uncommitted篇(转)
本文转自https://m.imooc.com/article/details?article_id=17291,感谢作者 1.目的 1.1 合适人群 1.数据库事务特征我只是背过,并没有很深刻的理解 ...
- C++封装随笔
1接口的封装和设计思想入门 接口的封装和设计思想入门 第一套api函数 #ifndef _CLT_SOCKET_H__ #define _CLT_SOCKET_H__ //客户端初始化环境 int c ...
- Paper | Predicting the Quality of Images Compressed After Distortion in Two Steps
目录 1. 问题本质剖析 2. 方法细节 图像质量评估大佬AC Bovik的论文,发表在2019 TIP上. 考虑的问题:对于有参考图像质量评估(R-IQA)任务,参考图像有时是有损的.这会导致评估的 ...
- SpringAOP ApectJ 动态代理
参考链接:https://docs.spring.io/spring/docs/4.3.13.RELEASE/spring-framework-reference/htmlsingle/#aop ht ...
随机推荐
- 【BZOJ】2002: [Hnoi2010]Bounce 弹飞绵羊(lct)
http://www.lydsy.com/JudgeOnline/problem.php?id=2002 (BZOJ挂了,还没在BZOJ测,先是在wikioi测过了,,) 囧.在军训时立志要学lct! ...
- JAVA生成RSA非对称型加密的公钥和私钥(利用JAVA API)
非对称型加密非常适合多个客户端和服务器之间的秘密通讯,客户端使用同一个公钥将明文加密,而这个公钥不能逆向的解密,密文发送到服务器后有服务器端用私钥解密,这样就做到了明文的加密传送. 非对称型加密也有它 ...
- OpenCV IplImage FlyCapture2 Image Conversion 两种图像类的相互转化
OpenCV的IplImag和 FlyCapture2 的 Image是两种常见的图片格式,在实际的应用中,我们通常要混合使用OpenCV和FlyCapture2这两个SDK,所以这两种图片格式之间的 ...
- FlyCapture2 fc2Image OpenCV IplImage Conversion 两种图像格式之间的转换
fc2Image是FlyCapture SDK的C语言库中的图片格式,由于在Windows上的MinGW无法编译FlyCapture2的C++库,只能使用C语言库,所以当我们在同时使用OpenCV的图 ...
- thinkphp 代码执行
相关漏洞:http://loudong.360.cn/vul/info/id/2919 ThinkPHP 开启lite模式后,会加载ThinkPHP/Extend/Mode/Lite/Dispache ...
- workerman是一个高性能的PHP socket服务器框架
workerman-chatorkerman是一款纯PHP开发的开源高性能的PHP socket服务器框架.被广泛的用于手机app.手游服务端.网络游戏服务器.聊天室服务器.硬件通讯服务器.智能家居. ...
- 通过Sysprep封装系统
<?xml version="1.0" encoding="utf-8"?> <unattend xmlns="urn:schema ...
- 《GK101任意波发生器》升级固件发布(版本:1.0.2.build124)
一.固件说明: 硬件版本:0,logic.3 固件版本:1.0.2.build124 编译日期:2014-08-19 ====================================== 二. ...
- Scrum会议3
组名称:天天向上 项目名称:连连看 参会成员:王森(Master)张金生 张政 栾骄阳 时间:2016.10.18 已完成内容: 1.GUI布局设计 2.通过在网上大量阅览代码,大体了解连连看游戏制作 ...
- var object dynamic的区别
一.var var本身不是一种类型,只是一种语法糖:var声明的变量在赋值的时候即已决定其变量类型,编译时会进行校验. 二.object object是所以类型的基类,故可以赋任何类型的值. 三.dy ...