work hard work smart

专注于Java后端开发。 不断总结,举一反三。
  博客园  :: 首页  :: 新随笔  :: 联系 :: 订阅 订阅  :: 管理

Spring AOP原理

Posted on 2019-03-15 18:26  work hard work smart  阅读(300)  评论(0编辑  收藏  举报

一、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字节码技术虚拟生成代理类)