Jdk动态代理为啥不能代理Class?

简介: 该文章主要介绍了JDK动态代理的原理以及为何JDK动态代理不能代理Class。

看完了SpringAop,明白了Spring底层封装了cglib和jdk动态代理,并且默认使用了Jdk动态代理来实现aop技术,可以说没有动态代理就没有SpringAop,这下好了,Spring必须叫Jdk动态代理为大哥!!

image.png

既然jdk动态代理地位如此重要,那么Jdk动态代理到底是如何实现的?代理类到底是如何生成的?

image.png

咱们还是先回顾下动态代理的使用姿势吧,先以LoginService接口的login方法为例

1、Jdk动态代理使用步骤一实现代理逻辑

实现InvocationHandler接口,实现相关代理逻辑

//实现invoke方法,实现代理逻辑
class ProxyInvocationHandler implements InvocationHandler {
   
   

    //原对象-被代理对象
    private Object source;

    public ProxyInvocationHandler(Object source) {
   
   
        this.source = source;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
   
   
        System.out.println("代理类:"+ proxy.getClass() + "被代理类:" + source.getClass());
        System.out.println("代理开始执行方法=====" + method.getName());
        Object result = method.invoke(source, args);
        System.out.println("代理结束执行方法=====" + method.getName());
        return result;
    }
}

2、Jdk动态代理使用步骤二创建代理对象

public static LoginService createProxy() {
   
   
    LoginService loginService = (LoginService) Proxy.newProxyInstance(LoginService.class.getClassLoader(), new Class[]{
   
   LoginService.class}, new ProxyInvocationHandler(new LoginServiceImpl()));
    System.out.println(loginService.getClass().toGenericString());
    return loginService;
}

3、Jdk动态代理使用步骤三使用代理对象

public static void main(String[] args) throws InterruptedException {
   
   
    //使用代理,和原对象一样的使用
    LoginService loginServiceProxy = createProxy();
    loginServiceProxy.login("周杰伦");
}

执行代理逻辑

image.png 上面完整演示了使用jdk动态代理过程。

image.png

Jdk动态代理使用起来还是很方便的,使用代理类和被代理类一样,生成代理类有两个重要的api,一个是InvocationHandler,一个是Proxy类

  • InvocationHandler接口比较简单,里面只有一个invoke方法,实现类需要处理代理逻辑。
  • 另外一个Proxy就是重点了,如果要弄明白代理到底是如何实现的,必须看看Proxy.newProxyInstance的源码。

image.png

Proxy.newProxyInstance的源码

public static Object newProxyInstance(ClassLoader loader,
                                          Class<?>[] interfaces,
                                          InvocationHandler h)
    {
   
   
        //1、获取被代理类的接口
        final Class<?>[] intfs = interfaces.clone();
        //2、生成代理类
        Class<?> cl = getProxyClass0(loader, intfs);
        //3、获取构造函数
        final Constructor<?> cons = cl.getConstructor(constructorParams);
        //4、代理逻辑类
        final InvocationHandler ih = h;
        //5、创建代理类的实例
        return cons.newInstance(new Object[]{
   
   h});   
    }

方法逻辑非常清晰,主要是通过java.lang.reflect.Proxy#getProxyClass0方法生成了一个类,这个就是代理类,但是我们看不到它生成的类具体是什么样子的, 在生成代理类的方法中,jdk开发者给我们留了一个开关,这个开关可以配置是否保存代理类的class文件。

生成代理class文件

//获取保存代理类class的开关
boolean saveGeneratedFiles = (Boolean)AccessController.doPrivileged(new GetBooleanAction("sun.misc.ProxyGenerator.saveGeneratedFiles"));

public static byte[] generateProxyClass(final String var0, Class<?>[] var1, int var2) {
   
   
    ProxyGenerator var3 = new ProxyGenerator(var0, var1, var2);
    final byte[] var4 = var3.generateClassFile();
    //开关打开了
    if (saveGeneratedFiles) {
   
   
        int var1 = var0.lastIndexOf(46);
        Path var2;
        //将代理类写到文件中
        Files.write(var2, var4, new OpenOption[0]);
        return null;
    }
    return var4;
}

那么我们可以配置系统参数sun.misc.ProxyGenerator.saveGeneratedFiles,打开这个开关

public static void main(String[] args) throws InterruptedException {
   
   

    //打开保存生成代理类开关
    System.getProperties().put("sun.misc.ProxyGenerator.saveGeneratedFiles", "true");
    LoginService loginServiceProxy = createProxy();
    loginServiceProxy.login("周杰伦");
}

这样运行我们的测试类后,会将代理类的class文件生成到com.sun.proxy包中: image.png

image.png

查看代理类class文件

public final class $Proxy0 extends Proxy implements LoginService {
   
   
    private static Method m1;
    private static Method m3;
    private static Method m2;
    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});
        } catch (RuntimeException | Error var3) {
   
   
            throw var3;
        } catch (Throwable var4) {
   
   
            throw new UndeclaredThrowableException(var4);
        }
    }

    public final void login(String var1) throws  {
   
   
        try {
   
   
            super.h.invoke(this, m3, new Object[]{
   
   var1});
        } 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 int hashCode() throws  {
   
   
        try {
   
   
            return (Integer)super.h.invoke(this, m0, (Object[])null);
        } catch (RuntimeException | Error var2) {
   
   
            throw var2;
        } catch (Throwable var3) {
   
   
            throw new UndeclaredThrowableException(var3);
        }
    }

    static {
   
   
        try {
   
   
            m1 = Class.forName("java.lang.Object").getMethod("equals", Class.forName("java.lang.Object"));
            m3 = Class.forName("cn.supfox.proxy.service.LoginService").getMethod("login", Class.forName("java.lang.String"));
            m2 = Class.forName("java.lang.Object").getMethod("toString");
            m0 = Class.forName("java.lang.Object").getMethod("hashCode");
        } catch (NoSuchMethodException var2) {
   
   
            throw new NoSuchMethodError(var2.getMessage());
        } catch (ClassNotFoundException var3) {
   
   
            throw new NoClassDefFoundError(var3.getMessage());
        }
    }
}

代理类总结

  1. 代理类被final修饰,类名称是$Proxy开头,数字结尾
  2. 代理类都继承了Proxy类
  3. 实现了被代理类的所有方法并且都是final修饰
  4. 实现的被代理类的方法,里面都是通过invocation.invoke方法触发方法调用,再看看咱们写的ProxyInvocationHandler类,这就能明白了代理类会通过invoke方法来代理具体的方法
//实现invoke方法,实现代理逻辑
class ProxyInvocationHandler implements InvocationHandler {
   
   

    //原对象-被代理对象
    private Object source;

    public ProxyInvocationHandler(Object source) {
   
   
        this.source = source;
    }

    @Override
    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
   
   
        System.out.println("代理类:"+ proxy.getClass() + "被代理类:" + source.getClass());
        System.out.println("代理开始执行方法=====" + method.getName());
        Object result = method.invoke(source, args);
        System.out.println("代理结束执行方法=====" + method.getName());
        return result;
    }
}

代理类思考总结

  1. jdk动态代理为啥只能代理接口不能代理普通class类?,因为代理对象已经实现了Proxy类,没法实现其他类了。
  2. 代理类通过持有ProxyInvocationHandler的引用,间接实现代理逻辑的功能。

image.png

相关文章
|
3月前
|
Java Spring
JDK动态代理和CGLIB动态代理的区别
Spring AOP中的动态代理主要有两种方式,JDK动态代理和CGLIB动态代理: ● JDK动态代理只提供接口的代理,不支持类的代理Proxy.newProxyInstance(类加载器, 代理对象实现的所有接口, 代理执行器) ● CGLIB是通过继承的方式做的动态代理 , 如果某个类被标记为final,那么它是无法使用 CGLIB做动态代理的。Enhancer.create(父类的字节码对象, 代理执行器)
|
3月前
|
监控 Java API
JDK动态代理和CGLIB动态代理
Java动态代理允许在运行时创建代理对象,增强或拦截目标类方法的执行。主要通过两种方式实现:JDK动态代理和CGLIB动态代理。JDK动态代理基于接口,利用`java.lang.reflect.Proxy`类和`InvocationHandler`接口;CGLIB则通过字节码技术生成目标类的子类作为代理,适用于未实现接口的类。两者均用于在方法执行前后添加额外逻辑,如日志记录、权限控制等,广泛应用于AOP框架中。
104 2
|
4月前
|
监控 Java API
JDK动态代理和CGLIB动态代理
Java动态代理允许在运行时创建代理对象,增强或拦截目标类的方法调用,无需修改原代码。它有两种主要实现方式:JDK动态代理和CGLIB动态代理。 - **JDK动态代理**:通过`java.lang.reflect.Proxy`类和`InvocationHandler`接口实现,适用于实现了接口的类。它在方法调用前后插入额外逻辑,如日志记录、权限控制等。 - **CGLIB动态代理**:基于字节码技术,为未实现接口的类生成子类作为代理,重写父类方法以添加增强逻辑。适用于没有接口的类,但要求目标类不能是`final`类或方法。
|
4月前
|
Java API 数据安全/隐私保护
探索Java动态代理的奥秘:JDK vs CGLIB
动态代理是一种在 运行时动态生成代理类的技术,无需手动编写代理类代码。它通过拦截目标方法的调用,实现对核心逻辑的 无侵入式增强(如日志、事务、权限控制等)。
112 0
探索Java动态代理的奥秘:JDK vs CGLIB
|
7月前
|
安全 Java 开发者
AOP中的JDK动态代理与CGLIB动态代理:深度解析与实战模拟
【11月更文挑战第21天】面向切面编程(AOP,Aspect-Oriented Programming)是一种编程范式,它通过将横切关注点(cross-cutting concerns)与业务逻辑分离,以提高代码的可维护性和可重用性。在Java开发中,AOP的实现离不开动态代理技术,其中JDK动态代理和CGLIB动态代理是两种常用的方式。本文将从背景、历史、功能点、业务场景、底层逻辑等多个维度,深度解析这两种代理方式的区别,并通过Java示例进行模拟和比较。
510 5
|
8月前
|
Java
【编程进阶知识】静态代理、JDK动态代理及Cglib动态代理各自存在的缺点及代码示例
本文介绍了三种Java代理模式:静态代理、JDK动态代理和Cglib动态代理。静态代理针对特定接口或对象,需手动编码实现;JDK动态代理通过反射机制实现,适用于所有接口;Cglib动态代理则基于字节码技术,无需接口支持,但需引入外部库。每种方法各有优缺点,选择时应根据具体需求考虑。
98 1
|
10月前
|
开发者 C# 容器
【独家揭秘】当WPF邂逅DirectX:看这两个技术如何联手打造令人惊艳的高性能图形渲染体验,从环境搭建到代码实践,一步步教你成为图形编程高手
【8月更文挑战第31天】本文通过代码示例详细介绍了如何在WPF应用中集成DirectX以实现高性能图形渲染。首先创建WPF项目并使用SharpDX作为桥梁,然后在XAML中定义承载DirectX内容的容器。接着,通过C#代码初始化DirectX环境,设置渲染逻辑,并在WPF窗口中绘制图形。此方法适用于从简单2D到复杂3D场景的各种图形处理需求,为WPF开发者提供了高性能图形渲染的技术支持和实践指导。
758 0
|
1月前
|
Java 关系型数据库 MySQL
在Linux平台上进行JDK、Tomcat、MySQL的安装并部署后端项目
现在,你可以通过访问http://Your_IP:Tomcat_Port/Your_Project访问你的项目了。如果一切顺利,你将看到那绚烂的胜利之光照耀在你的项目之上!
176 41
|
1月前
|
开发框架 Java 关系型数据库
在Linux系统中安装JDK、Tomcat、MySQL以及部署J2EE后端接口
校验时,浏览器输入:http://[your_server_IP]:8080/myapp。如果你看到你的应用的欢迎页面,恭喜你,一切都已就绪。
243 17
|
2月前
|
Oracle Java 关系型数据库
Tomcat和JDK的详细安装、下载和环境配置指南
以上就是JDK和Tomcat的下载、安装和环境配置的详细步骤。希望这个指南能帮助你顺利完成设置。
182 32