博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Spring框架学习06——AOP底层实现原理
阅读量:5207 次
发布时间:2019-06-14

本文共 5457 字,大约阅读时间需要 18 分钟。

在Java中有多种动态代理技术,如JDK、CGLIB、Javassist、ASM,其中最常用的动态代理技术是JDK和CGLIB。

1、JDK的动态代理

JDK动态代理是java.lang.reflect.*包提供的方法,必须要借助一个接口才能产生代理对象,对于使用业务接口的类,Spring默认使用JDK动态代理实现AOP。

代码示例如下:
创建dao包,并创建StuDao接口和StuDaoImpl实现类,
StuDao接口

public interface StuDao {    public void add();    public void find();    public void update();    public void delete();}

StuDaoImpl实现类

public class StuDaoImpl implements StuDao {    @Override    public void add() {        System.out.println("添加学生");    }    @Override    public void find() {        System.out.println("查询学生");    }    @Override    public void update() {        System.out.println("修改学生");    }    @Override    public void delete() {        System.out.println("删除学生");    }}

创建aspect包,并创建切面类MyAspect,该类中可以定义多个通知,即增强处理的方法,示例代码如下:

public class MyAspect {    public void check(){        System.out.println("模拟权限控制");    }    public void except(){        System.out.println("模拟异常处理");    }    public void log(){        System.out.println("模拟日志记录");    }    public void monitor(){        System.out.println("模拟性能检测");    }}

创建proxy包,并创建代理类MyJdkProxy,在JDK动态代理中代理类必须实现java.lang.reflect.InvocationHandler接口,并编写代理方法,在代理方法中需要通过Proxy实现动态代理。示例代码如下:

package com.aop.proxy;import com.aop.aspect.MyAspect;import com.aop.dao.StuDao;import java.lang.reflect.InvocationHandler;import java.lang.reflect.Method;import java.lang.reflect.Proxy;public class MyJdkProxy implements InvocationHandler {    //声明目标类接口对象(真实对象)    private StuDao stuDao;    public MyJdkProxy(StuDao stuDao){        this.stuDao = stuDao;    }    //创建代理的方法,建立代理对象和真实对象的代理关系,返回代理对象    public Object createProxy(){        //1.类加载器        ClassLoader cld = MyJdkProxy.class.getClassLoader();        //2.被代理对象实现的所有接口        Class[] clazz = stuDao.getClass().getInterfaces();        return Proxy.newProxyInstance(cld,clazz,this);    }    /**     * 代理的逻辑方法,所有动态代理类的方法调用都交给该方法处理     * @param proxy 被代理对象     * @param method 要执行的方法     * @param args 执行方法时需要的参数     * @return 返回代理结果     * @throws Throwable     */    @Override    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {        //创建一个切面        MyAspect myAspect = new MyAspect();        //前增强        myAspect.check();        myAspect.except();        //在目标类上调用方法并传入参数,相当于调用stuDao中的方法        Object obj = method.invoke(stuDao,args);        //后增强        myAspect.log();        myAspect.monitor();        return obj;    }}

创建测试类

@Testpublic void testStu(){    //创建目标对象    StuDao stuDao = new StuDaoImpl();    //创建代理对象    MyJdkProxy myJdkProxy = new MyJdkProxy(stuDao);    //从代理对象中获取增强后的目标对象    //该对象是一个被代理的对象,它会进入代理的逻辑方法invoke中    StuDao stuDaoProxy = (StuDao) myJdkProxy.createProxy();    //执行方法    stuDaoProxy.add();    System.out.println("==================");    stuDaoProxy.update();    System.out.println("==================");    stuDaoProxy.delete();}

运行结果

2、CGLIB的动态代理

JDK动态代理必须提供接口才能使用,对于没有提供接口的类,只能采用CGLIB动态代理。CGLIB采用非常底层的字节码技术,对指定的目标类生产一个子类,并对子类进行增强。在Spring Core 包中已经集成了CGLIB所需要的jar包,无需另外引入jar包。

示例代码如下:
创建目标类TestDao

public class TestDao {    public void save(){        System.out.println("保存方法");    }    public void modify(){        System.out.println("修改方法");    }    public void delete(){        System.out.println("删除方法");    }}

创建切面类MyAspect,并在该类中定义多个通知

public class MyAspect {    public void check(){        System.out.println("模拟权限控制");    }    public void except(){        System.out.println("模拟异常处理");    }    public void log(){        System.out.println("模拟日志记录");    }    public void monitor(){        System.out.println("模拟性能检测");    }}

创建代理类MyCglibProxy,并实现MethodInterceptor接口

package com.aop.proxy;import com.aop.aspect.MyAspect;import org.springframework.cglib.proxy.Enhancer;import org.springframework.cglib.proxy.MethodInterceptor;import org.springframework.cglib.proxy.MethodProxy;import java.lang.reflect.Method;public class MyCglibProxy implements MethodInterceptor {    /**     * 创建代理的方法,生成CGLIB代理对象     * @param target 目标对象,需要增强的对象     * @return 返回目标对象的CGLIB代理对象     */    public Object createProxy(Object target){        //创建一个动态类对象,即增强类对象        Enhancer enhancer = new Enhancer();        //设置其父类        enhancer.setSuperclass(target.getClass());        //确定代理逻辑对象为当前对象        enhancer.setCallback(this);        return enhancer.create();    }    /**     * 该方法会在程序执行目标方法时调用     * @param proxy 是CGLIB根据指定父类生成的代理对象     * @param method 是拦截方法     * @param args 拦截方法的参数数组     * @param methodProxy 方法的代理对象,用于执行父类的方法     * @return 返回代理结果     * @throws Throwable     */    @Override    public Object intercept(Object proxy, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {        //创建一个切面        MyAspect myAspect = new MyAspect();        //前置增强        myAspect.check();        //目标方法执行,返回执行结果        Object obj = methodProxy.invokeSuper(proxy,args);        //后置增强        myAspect.log();        return obj;    }}

创建测试类

@Testpublic void test(){    //创建目标对象    TestDao testDao = new TestDao();    //创建代理对象    MyCglibProxy myCglibProxy = new MyCglibProxy();    //获取增强后的目标对象    TestDao testDaoAdvice = (TestDao) myCglibProxy.createProxy(testDao);    //执行方法    testDaoAdvice.save();    System.out.println("==================");    testDaoAdvice.modify();    System.out.println("==================");    testDaoAdvice.delete();}

运行结果

3、动态代理注意事项

(1)程序中应优先对接口创建代理,便于程序解耦维护;

(2)使用final关键字修饰的方法不能被代理,因为无法覆盖

  • JDK动态代理,是针对接口生成子类,接口中方法不能使用final修饰
  • CGLIB是针对目标类生成子类,因此类或方法不能使用final修饰

(3)Spring只支持方法连接点,不提供属性连接点

 

转载于:https://www.cnblogs.com/jpwz/p/10567384.html

你可能感兴趣的文章
链表快排
查看>>
链表操作合集
查看>>
leetcode852 C++ 20ms 找最高峰 序列先增后减
查看>>
JavaScript深入系列(一)--原型和原型链详解
查看>>
一点感想
查看>>
产生随机数
查看>>
vm center(VC)5.1登陆密码忘记了怎么办?
查看>>
TFS 2015 敏捷开发实践 – 看板的使用
查看>>
UINavigationController的简单使用
查看>>
Python命名规范
查看>>
50款漂亮的国外婚礼邀请函设计(上篇)
查看>>
MD5加密简单算法
查看>>
安装Qcreator2.5 + Qt4.8.2 + MinGW_gcc_4.4 (win7环境)
查看>>
代码检查
查看>>
滚动条
查看>>
程序员的自我修养九Windows下的动态链接
查看>>
记一次修改数据库引擎的方法
查看>>
开发工具 idea 激活方法
查看>>
BZOJ 4052: [Cerc2013]Magical GCD
查看>>
libevent和libcurl的一些学习
查看>>