一、为什么需要AOP技术
AOP 是一个很成熟的技术。
假如项目中有方法A、方法B、方法C……等多个方法,
如果项目需要为方法A、方法B、方法C……这批方法增加具有通用性质的横切处理。
下图可以形象的说明具有通用性质的横切处理的思想:
在以前传统的做法是
- 先定义一个Advice方法,该方法实现这个通用性质的横切处理。
- 打开方法A、方法B、方法C……的源代码修改,使得方法A、方法B、方法C……去调用Advice方法。
客户电话: 为每个方法都增加日志。 客户电话: 为每个方法前都增加权限控制。 客户电话: 为每个方法都加…… ….
如果使用AOP,可以做到程序员无需修改方法A、方法B、方法C……,但又可以为方法A、方法B、方法C增加调用Advice方法。
面向切面编程(AOP)是作为面向对象编程(OOP)的补充 AOP框架具有如下两个特征:
- 各步骤之间的良好隔离性。
- 源代码无关性。
二、AOP的功能
保证程序员不修改方法A、方法B、方法C……的前提下,可以为方法A、方法B、方法C……增加通用处理。
AOP的本质:依然要去【修改】方法A、方法B、方法C……
—— 只是这个修改由AOP框架完成,程序员不需要改。
AOP要求去修改,到底怎么去修改方法A、方法B、方法……
AOP的实现方式有两种
AOP框架在编译阶段,就对目标类进行修改,得到的class文件已经是被修改过的。生成静态的AOP代理类(生成*.class文件已经被改掉了,需要使用特定的编译器)。以AspectJ为代表 —— 静态AOP框架。
AOP框架在运行阶段,动态生成AOP代理(在内存中动态地生成AOP代理类),以实现对目标对象的增强。它不需要特殊的编译器。以Spring AOP为代表。—— 动态AOP框架。
上面两种,哪种性能更好?很明显静态的AOP框架更好。
下面我们进入AspectJ的学习
三、实战AspectJ
AspectJ是一个基于Java语言的AOP框架,提供了强大的AOP功能,其他很多AOP框架都借鉴或采纳其中的一些思想。
下载和安装AspectJ
- 运行、下载得到的安装JAR包。在命令行窗口启动下载得到的jar文件:java -jar aspectj-1.6.10.jar,在弹出的安装界面会先让你选择Java,选择你安装的Java目录 即可。将该软件(绝对绿色)安装到指定目录下(笔者安装在C盘)。
- 系统还应该将
C:Javaaspectj1.6bin
路径添加到PATH环境变量中,将C:Javaaspectj1.6libaspectjrt.jar和aspectjtools.jar添加到 CLASSPATH环境变量中。
成功安装了AspectJ之后,将会在E:JavaAOPaspectj1.6
路径下(AspectJ的安装路径)看到如下文件结构:
- bin:该路径下存放了aj、aj5、ajc、ajdoc、ajbrowser等命令,其中ajc命令最常用,它的作用类似于javac,用于对普通Java类进行编译时增强。
- docs:该路径下存放了AspectJ的使用说明、参考手册、API文档等文档。
- lib:该路径下的4个JAR文件是AspectJ的核心类库。
- 相关授权文件。
打开DOS窗口,输入ajc命令,可以看到输出ajc命令的所有信息,即可知安装和环境变量配置成功:
使用AspectJ
接下来我们模拟一个普通程序: UserService:
代码语言:javascript复制package com.mybry.aop.service;
public class UserService{
public int addUser(){
System.out.println("模拟添加用户的方法。");
return 20;
}
public void validateLogin(){
System.out.println("验证用户登录。");
}
}
BookServce:
代码语言:javascript复制package com.mybry.aop.service;
public class BookService{
public int addBook(String name,int price){
System.out.println("正在添加图书,书名是:" name ",价格是:" price);
return 100;
}
}
编译运行结果:
这两个类正好相当于我们的方法A,方法B…..
假如客户现在要求在每个方法前面增加权限检查功能,那么该如何做呢?下面我们就是用AspectJ来实现这个功能。
现在我们在这个模拟程序基础上增加这个AOP功能
我们先写一个权限检查的Aspectj类:
实例1,AuthAspect:
代码语言:javascript复制package com.mybry.aop.aspectj;
public aspect AuthAspect{
// Advice
// execution(* com.mybry.aop.service.*.*(..)执行 任意返回值 改包下的任意类的任意方法形参不限
before():execution(* com.mybry.aop.service.*.*(..)){
// 对原来方法进行修改、增强。
System.out.println("----------模拟执行权限检查----------");
}
}
注意这个类色声明类型:aspect,没错,这是写Aspect必须声明的类型,只有AspectJ编译器可以识别。
再用ajc –d *.java
命令编译执行:
太开心了,搞定了!
假如此时客户又要求在每个方法中增加事物处理呢?也好办,下面是事物处理类: 实例1,TxAspect:
代码语言:javascript复制package com.mybry.aop.aspectj;
public aspect TxAspect{
//around的意思就是在方法的前面和后面都加
Object around():execution(* com.mybry.aop.service.*.*(..)){
// 对原来方法进行修改、增强。
System.out.println("====模拟开启事务====");
Object rvtVal = proceed();
System.out.println("====模拟结束事务====");
return rvtVal;
}
}
好了,我们再来编译运行它:
OK!我们的Aspect AOP程序到此为止!