彻底搞懂JDK动态代理

通过一张图,先了解什么是代理模式

可以看到这种模式可以在调用方与目标对象之间起到中介作用,从而保护目标对象,并可以扩展目标对象的功能(ps:这里可以想想符合一种什么设计原则

JDK动态代理具体实现

JDK动态代理是基于接口的代理,只能对实现了接口的类进行代理

1.定义接口,实现动态代理的目标接口

java 复制代码
//定义接口
public interface TargetInterFace {

    void targetMethod();
}
java 复制代码
//接口实现
public class TargetObject implements TargetInterFace{

    @Override
    public void targetMethod() {
        System.out.println("/// "+this.getClass().getName()+" 执行");
    }
}

2.实现InvocationHandler接口,重写invoke方法

InvocationHandler是Java中的一个接口,是Java反射API的一部分,它用于动态创建接口的代理实例。

java 复制代码
public class AopInvocation implements InvocationHandler {

    private Object target;

    public AopInvocation(Object target) {
        this.target = target;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {

        System.out.println("/// "+this.getClass().getName()+" 执行前");

        method.invoke(this.target, args);

        System.out.println("/// "+this.getClass().getName()+" 执行后");

        return null;
    }
}

InvocationHandler的主要作用是将方法调用与具体的实现解耦,使得在代理对象中可以灵活地处理方法调用。

这个接口只有一个方法,即invoke()方法,它负责处理对代理对象的方法调用(当代理实例调用方法时,方法调用被编码分派到InvocationHandler的invoke方法)

通过使用InvocationHandler,可以更加灵活地扩展程序的功能(例如在调用前后添加额外的逻辑、对调用参数进行校验等),而不需要修改原有的代码

3.创建动态代理对象,将目标对象和InvocationHandler传递给Proxy.newProxyInstance方法。

Proxy类是JDK提供的一个工具类,用于创建代理对象。它提供了一个静态方法newProxyInstance(ClassLoader loader, Class<?>[] interfaces, InvocationHandler h),用于创建代理对象。该方法需要三个参数:

  • ClassLoader loader:指定代理对象的类加载器。
  • Class<?>[] interfaces:指定代理对象要实现的接口。
  • InvocationHandler h:指定实现了InvocationHandler接口的对象,用于处理方法调用。
java 复制代码
public static void main(String[] args) {
    //目标对象
    TargetObject targetObject = new TargetObject();
    //建立代理与被代理对象的关系
    AopInvocation aopInvocation=new AopInvocation(targetObject);
    //生成代理对象
    TargetInterFace instance = (TargetInterFace) Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), aopInvocation);
    instance.targetMethod();
}

JDK动态代理原理

为什么没有显式调用invoke,但invoke方法确实被执行了?

通过Proxy.newProxyInstance方法为大家解释

swift 复制代码
public static Object newProxyInstance(ClassLoader loader,
                                      Class<?>[] interfaces,
                                      InvocationHandler h)
    throws IllegalArgumentException
{
    /**
    * 非核心代码省略
    */
    
    /*
     * 获取到代理接口的class
     */
    Class<?> cl = getProxyClass0(loader, intfs);
    
    /*
     * Invoke its constructor with the designated invocation handler.
     */
    try {
        /**
        * 用反射获取构造方法,然后创建代理类实例
        */
        final Constructor<?> cons = cl.getConstructor(constructorParams);
        final InvocationHandler ih = h;
        
      /**
        * 非核心代码省略
       */
        return cons.newInstance(new Object[]{h});
    } catch (IllegalAccessException|InstantiationException e) {
        throw new InternalError(e.toString(), e);
    }
}

通过getProxyClass 获取到代理接口的class($Proxy0)

通过Arthas反编译$Proxy0的源码,可以看到该类继承了目标接口

然后通过用反射获取构造方法,然后创建代理类实例。

代理对象的方法调用都是通过super.h.invoke(this,m1,(Object[])null)调用,其中的 super.h.invoke实际上是在创建代理的时候传递给Proxy.newProxyInstance的 代理对象,它继承InvocationHandler类,负责实际的调用处理逻辑。 代理对象invoke()接收到method、args等参数后,然后通过反射让被代理的对象执行目标方法

总结

JDK动态代理符合我们的开闭原则。

JDK动态代理主要面向接口,可以为接口中的所有方法提供代理,没有实现接口的类无法进行代理。

相关推荐
Lionel_SSL3 小时前
《深入理解Java虚拟机》第三章读书笔记:垃圾回收机制与内存管理
java·开发语言·jvm
记得开心一点嘛3 小时前
手搓Springboot
java·spring boot·spring
老华带你飞4 小时前
租房平台|租房管理平台小程序系统|基于java的租房系统 设计与实现(源码+数据库+文档)
java·数据库·小程序·vue·论文·毕设·租房系统管理平台
独行soc4 小时前
2025年渗透测试面试题总结-66(题目+回答)
java·网络·python·安全·web安全·adb·渗透测试
脑子慢且灵4 小时前
[JavaWeb]模拟一个简易的Tomcat服务(Servlet注解)
java·后端·servlet·tomcat·intellij-idea·web
华仔啊5 小时前
SpringBoot 中 6 种数据脱敏方案,第 5 种太强了,支持深度递归!
java·后端
异常驯兽师6 小时前
Spring 中处理 HTTP 请求参数注解全解析
java·spring·http
连合机器人7 小时前
晨曦中的守望者:当科技为景区赋予温度
java·前端·科技
AD钙奶-lalala7 小时前
idea新建的项目new 没有java class选项
java·ide·intellij-idea
sheji34167 小时前
【开题答辩全过程】以 12306候补购票服务系统为例,包含答辩的问题和答案
java·eclipse