AOP 是一个很成熟的技术。
假如项目中有方法A、方法B、方法C……等多个方法,
如果项目需要为方法A、方法B、方法C……这批方法增加具有通用性质的横切处理。
下图可以形象的说明具有通用性质的横切处理的思想:
在以前传统的做法是
客户电话: 为每个方法都增加日志。 客户电话: 为每个方法前都增加权限控制。 客户电话: 为每个方法都加…… ….
如果使用AOP,可以做到程序员无需修改方法A、方法B、方法C……,但又可以为方法A、方法B、方法C增加调用Advice方法。
面向切面编程(AOP)是作为面向对象编程(OOP)的补充 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是一个基于Java语言的AOP框架,提供了强大的AOP功能,其他很多AOP框架都借鉴或采纳其中的一些思想。
下载和安装AspectJ
C:\Java\aspectj1.6\bin
路径添加到PATH环境变量中,将C:\Java\aspectj1.6\lib\aspectjrt.jar和aspectjtools.jar添加到 CLASSPATH环境变量中。成功安装了AspectJ之后,将会在E:\Java\AOP\aspectj1.6
路径下(AspectJ的安装路径)看到如下文件结构:
打开DOS窗口,输入ajc命令,可以看到输出ajc命令的所有信息,即可知安装和环境变量配置成功:
使用AspectJ
接下来我们模拟一个普通程序: UserService:
package com.mybry.aop.service;
public class UserService{
public int addUser(){
System.out.println("模拟添加用户的方法。");
return 20;
}
public void validateLogin(){
System.out.println("验证用户登录。");
}
}
BookServce:
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:
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:
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程序到此为止!