前言:
学习过spring的都知道,IOC和AOP是spring里最基础的两个核心技术。
在学习AOP(面向切面编程)的时候用到了动态代理,因为AOP的底层实现原理使用的就是动态代理。spring默认使用的是jdk提供的动态代理,其实是两种都支持。


目录

  • **什么是动态代理?**
  • **代理模式:**
  • **JDKProxDynameic 和 CGlibProxDynameic区别**
  • 两种动态代理实现方式:
  • JDKProxDynameic(jdk提供)
  • 代码实现:
  • CGlibProxDynameic(spring提供)
  • 代码实现:

目录)

什么是动态代理?

在了解动态代理之前,我现在简单介绍下什么代理。代理:按照字面意思,就是代替你去做一些事情。代替你去完成一些功能,或者做一些本来应该你来做的事情。这是字面意思理解,在面向对象的程序设计语言里:动态代理是在你原有的功能基础之上,对功能进行增强的一种实现手段。通过动态代理,并且符合开闭原则的前提增强方法的功能。这也是AOP的思想,通过不修改原有代码,把你的代码织入到指定的方法中。

代理模式:

代理模式(Proxy):为其他对象提供一个代理以控制对这个对象的访问。
主要解决:在直接访问对象时带来的问题,比如说:要访问的对象在远程的机器上。在面向对象系统中,有些对象由于某些原因(比如对象创建开销很大,或者某些操作需要安全控制,或者需要进程外的访问),直接访问会给使用者或者系统结构带来很多麻烦,我们可以在访问此对象时加上一个对此对象的访问层。
代理模式的元素是:共同接口、代理对象、目标对象。
代理模式的行为:由代理对象执行目标对象的方法、由代理对象扩展目标对象的方法。
代理模式的宏观特性:对客户端只暴露出接口,不暴露它以下的架构。

JDKProxDynameic 和 CGlibProxDynameic区别

jdk的动态代理是代理的接口,并且只能代理有 接口的类。如果这个类没有实现任何接口,jdk的动态代理是无法代理的。
这个时候就可以使用cglib去对类的字节码进行底层的继承代理,通过继承被代理对象。也就是JDKProxDynameic 代理的接口 CGlibProxDynameic代理的类(如果类被final修饰就不能被代理成功)

两种动态代理实现方式:

JDKProxDynameic(jdk提供)

jdk的动态代理主要是:

InvocationHandler接口和proxy类

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_spring


import java.lang.reflect.InvocationHandler; 接口

InvocationHandler这个接口里主要是使用invoke方法,增强被代理对象的方法。

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_动态代理_02


参数:proxy 代理的实例

method 调用需要执行的方法

args 方法的参数import java.lang.reflect.Proxy; 代理类

proxy类使用newProxyInstance方法创建新的代理实例

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_cglib_03


参数:CLassLoader loader 代理对象的类加载器,用哪个类加载器去加载代理对象

Class<?> interfaces 被代理的接口数组

InvocationHandler h 调用处理器,也就是具体调用那个invoke方法。

代码实现:

被代理接口

package com.proxys.proxy;

/**
 * @author 康世行
 * @Title:
 * @Package com.kuang.proxy
 * @Description:
 * @date 2021-07-21 9:04
 */
public interface A {
   public void a();
   public void b();
}

被代理接口实现类

package com.proxys.proxy;

/**
 * @author 康世行
 * @Title:
 * @Package com.kuang.proxy
 * @Description:
 * @date 2021-07-21 14:46
 */
public class testimpl implements A{
    @Override
    public void a() {
        System.out.println("测试动态代理");
    }

    @Override
    public void b() {
        System.out.println("代理了b方法");
    }
}

代理对象

package com.proxys.proxy;

import com.kuang.springbootdemo.controller.Itest;

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;

/**
 * @author 康世行
 * @Title:
 * @Package com.kuang.proxy
 * @Description: jdk动态代理
 * @date 2021-07-21 14:28
 */
public class JdkDynamicProxyTest implements InvocationHandler {
    //被代理的接口
    private A a;
   //给具体的接口赋值
    public JdkDynamicProxyTest(A itest) {
      this.a=itest;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("执行了invoke方法");
        return method.invoke(a,args);//确定要执行那个对象的方法
    }
    //实例化动态代理
    public static A newProxyInstance(A itest){
        return (A) Proxy.newProxyInstance(JdkDynamicProxyTest.class.getClassLoader(),new Class[]{A.class},new JdkDynamicProxyTest(itest));
    }

}

测试代码

@Test
    void contextLoads() throws Exception {
       //创建测试对象(被代理对象)
        A a=new testimpl();
        //jdk动态代理实现,获取代理对象
        A jdkDynamicProxyTest = JdkDynamicProxyTest.newProxyInstance(a);
        jdkDynamicProxyTest.a();
        jdkDynamicProxyTest.b();
    }

测试结果:

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_cglib_04

CGlibProxDynameic(spring提供)

cglib动态代理主要是:

MethodInterceptor 方法拦截接口和Enhancer 增强类

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_jdk_05


MethodInterceptor 方法拦截接口里主要是使用 intercept方法对父方法进行拦截,然后增强父方法的功能。

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_动态代理_06


然后使用Enhancer增强类 里的create方法创建代理对象

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_动态代理_07

代码实现:

被代理对象使用上面JDKProxDynameic 里的A接口的实现类testimpl
代理对象

package com.kuang.proxy;

import org.springframework.cglib.proxy.Enhancer;
import org.springframework.cglib.proxy.MethodInterceptor;
import org.springframework.cglib.proxy.MethodProxy;

import java.lang.reflect.Method;

/**
 * @author 康世行
 * @Title:
 * @Package com.kuang.proxy
 * @Description: cglib动态代理
 * @date 2021-07-21 15:18
 */
public class cglibProxyTestt implements MethodInterceptor {
    @Override
    public Object intercept(Object o, Method method, Object[] objects, MethodProxy methodProxy) throws Throwable {
        System.out.println("intercept执行完毕");
        return methodProxy.invokeSuper(o,objects);
    }
    //产生cglibproxy动态代理(通过底层字节码继承,实现动态代理;注意,如果类被final修饰怎代理失败)
    public static <T extends A> A newProxyInstance(Class<T> targetInstanceClazz){
        Enhancer enhancer=new Enhancer();
        enhancer.setSuperclass(targetInstanceClazz);
        enhancer.setCallback(new cglibProxyTestt());
        return (A) enhancer.create();//创建动态代理
    }
}

测试代码

@Test
    void contextLoads() throws Exception {
       //创建测试对象(被代理对象)
        A a=new testimpl();
         
        // cglib创建动态代理对象
        A cglibProxyTest = cglibProxyTestt.newProxyInstance(testimpl.class);
        cglibProxyTest.a();
        cglibProxyTest.b();
    }

测试结果

SpringAop动态代理(proxy)两种实现方式 JDK 和CGlib_spring_08

感谢阅读 ~~~~