一、AOP介绍
AOP 全称Aspect Orient Programming,即面向切面编程,解决代码复用问题。是对OOP(Object Orient Programming)的一种补充。AOP实现的关键在于AOP框架自动创建的AOP代理,AOP代理分为静态代理和动态代理。
应该场景:广泛应用于处理一些具有横切性质的系统服务,如日志输出,安全控制,事务管理,缓存,对象池,异常处理等
1、静态代理:指使用AOP框架提供的命令进行编译,从而在编译阶段就生成AOP代理。也称编译时增强。
静态代理说白了,就是程序运行前就已经存在代理类的字节码文件、代理类和原始类的关系在运行前就已经确定了。
1) 静态代理Demo
IPerson接口
public interface IPerson {
//找工作
void findJob();
}
ZhangSan要去找工作
/**
* 目标对象
*/
public class ZhangSan implements IPerson {
@Override
public void findJob() {
System.out.println("我是张三,我在找工作...");
}
}
代理类:人才市场
public class PersonMarketProxy implements IPerson{
private IPerson target = new ZhangSan();
@Override
public void findJob() {
System.out.println("找工作前,请把简历给我");
target.findJob();
System.out.println("找到工作后,要好好工作");
}
}
测试
public class Main {
public static void main(String[] args) {
//代理对象
IPerson proxy = new PersonMarketProxy();
//代理对象执行代理方法
proxy.findJob();
}
}
静态代理虽然保证了业务。如果代理方法增多,势必每一个方法多要进行代理。除了实现类要实现这个方法,代理类也要实现这个方法。维护成本增加。
而动态代理很好的解决了这个问题。
2、动态代理:在运行时借助于JDK动态代理、CGLIG等内存中“临时”生成的AOP动态代理类。也称运行时增强。
动态代理Demo
2.1 IPerson接口
public interface IPerson {
//找工作
void findJob();
}
2.2 目标对象Zhangsan类
/**
* 目标对象
*/
public class ZhangSan implements IPerson {
@Override
public void findJob() {
System.out.println("我是张三,我在找工作...");
}
}
2.3 ProxyFactory 类
**
* 动态代理
* 给多个目标对象生成代理对象
*/
public class ProxyFactory {
//目标对象
private Object target;
public ProxyFactory(Object target){
this.target = target;
}
//返回代理对象
public Object getProxyObject(){
Object proxy = Proxy.newProxyInstance(
target.getClass().getClassLoader(), //目标对象类加载器
target.getClass().getInterfaces(), //目标对象实现的所有接口
new InvocationHandler() { //执行代理对象方法时候触发
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
String methodName = method.getName();
Object result = null;
if("findJob".equals(methodName)){
System.out.println("找工作前,请把简历给我");
result = method.invoke(target, args);
System.out.println("找到工作后,要好好工作");
}else{
result = method.invoke(target, args);
}
return result;
}
});
return proxy;
}
}
2.4 测试
public class Main {
public static void main(String[] args) {
//目标对象
IPerson target = new ZhangSan();
System.out.println("目标对象:" + target.getClass());
//代理对象
IPerson proxy = (IPerson)new ProxyFactory(target).getProxyObject();
System.out.println("代理对象:" + proxy.getClass());
//代理对象执行代理方法
proxy.findJob();
createProxyClassFile(IPerson.class);
}
private static void createProxyClassFile(Class c){
byte[] data = ProxyGenerator.generateProxyClass("$Proxy0", new Class[]{c});
try{
FileOutputStream fileOutputStream = new FileOutputStream("$Proxy0.class");
fileOutputStream.write(data);
fileOutputStream.close();
}catch (FileNotFoundException e){
e.printStackTrace();
}catch (IOException e){
e.printStackTrace();
}
}
}
2.5 打印结果:

IPerson proxy = (IPerson)new ProxyFactory(target).getProxyObject();
这行代码其实是JDK动态生成了一个类去实现接口
public final class $Proxy0 extends Proxy implements IPerson
完整的代码如2.6.
2.6 然后使用使用createProxyClassFile,将代理类使用写入文件,文件的内容为
//
// Source code recreated from a .class file by IntelliJ IDEA
// (powered by Fernflower decompiler)
//
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
import spring_springmvc.proxyDongTai.IPerson;
public final class $Proxy0 extends Proxy implements IPerson {
private static Method m1;
private static Method m2;
private static Method m3;
private static Method m0;
public $Proxy0(InvocationHandler var1) throws {
super(var1);
}
public final boolean equals(Object var1) throws {
try {
return ((Boolean)super.h.invoke(this, m1, new Object[]{var1})).booleanValue();
} catch (RuntimeException | Error var3) {
throw var3;
} catch (Throwable var4) {
throw new UndeclaredThrowableException(var4);
}
}
public final String toString() throws {
try {
return (String)super.h.invoke(this, m2, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final void findJob() throws {
try {
super.h.invoke(this, m3, (Object[])null);
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
public final int hashCode() throws {
try {
return ((Integer)super.h.invoke(this, m0, (Object[])null)).intValue();
} catch (RuntimeException | Error var2) {
throw var2;
} catch (Throwable var3) {
throw new UndeclaredThrowableException(var3);
}
}
static {
try {
m1 = Class.forName("java.lang.Object").getMethod("equals", new Class[]{Class.forName("java.lang.Object")});
m2 = Class.forName("java.lang.Object").getMethod("toString", new Class[0]);
m3 = Class.forName("spring_springmvc.proxyDongTai.IPerson").getMethod("findJob", new Class[0]);
m0 = Class.forName("java.lang.Object").getMethod("hashCode", new Class[0]);
} catch (NoSuchMethodException var2) {
throw new NoSuchMethodError(var2.getMessage());
} catch (ClassNotFoundException var3) {
throw new NoClassDefFoundError(var3.getMessage());
}
}
}
2.7 使用JDK生成动态代理的前提示目标类必须有实现的接口。 CGLIB是以动态生成的子类继承目标方式实现。在运行期动态的内存中构建一个子类
AOP是基于动态代理的
AOP是方法级别的
AOP可以分离业务和重复代码
二、实现AOP的技术
主要实现AOP思想的技术有AspectJ和Spring AOP
1、AspectJ的底层技术
AspectJ的底层技术是静态代理,即用一种AspectJ支持的特定语言编写切面,通过一个命令来编译,生成一个新的代理类。写着是编译时增强,相对于运行时增强性能更好。
2、Spring AOP
Spring AOP采用动态代理,在运行期间对业务方法进行增强,不会生成新类。Spring AOP提供了对JDK动态代理支持和CGLib的支持。
JDK动态代理只能为接口创建动态代理实例,而不能对类创建动态代理。需要获得被目标类的接口信息(使用反射技术),生成一个实现了代理接口的动态代理类(字节码),再通过反射机制获得动态代理的构造函数,利用构造函数生成代理类的实例对象,在调用具体方法前调用invokeHandler方法处理。
CGLib动态代理需要依赖asm包,把被代理对象类的class文件加载进来,修改其字节码生成子类。
但是Spring AOP基于注解配置的情况下,需要依赖于AspectJ包的标准注解,但是不需要额外的编译及AspectJ的织入器,而基于XML配置不需要。

Spring Aop过程
1、创建容器对象的时,根据切入点表达式拦截的类,生成代理对象
2、 如果有接口,使用JDK代理。反之,使用CGLIB代理。然后从容器中获取代理对象,在运行期间植入“切面” 类的方法。
如果目标类没有实现接口,且class为final, 则不能进行Spring AOP编程。
JDK动态代理和CGLib动态代理的区别
JDK动态代理需要接口,基于反射实现。(反射虚拟生成代理类)
CGLib动态代理需要子类实现。基于ASM字节码包装的一个类库。(基于ASM字节码技术虚拟生成代理类)