滚雪球学Java(25):动态代理

举报
bug菌 发表于 2024/01/30 12:16:21 2024/01/30
【摘要】 🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,助你一臂之力,带你早日登顶🚀,欢迎大家关注&&收藏!持续更新中,up!up!up!! 前言在Java开发中,经常会使用代理模式来实现一些特殊的需求,例如AOP编程、RPC调用等。Java中的代理模式主要有静态代理和动态代理两种,其中静态代理需要手动编写一个代理类,而动态代理则可以在运行时动态生成代理类,更加灵活方便。本文将介绍Ja...

🏆本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,助你一臂之力,带你早日登顶🚀,欢迎大家关注&&收藏!持续更新中,up!up!up!!


前言

在Java开发中,经常会使用代理模式来实现一些特殊的需求,例如AOP编程、RPC调用等。Java中的代理模式主要有静态代理和动态代理两种,其中静态代理需要手动编写一个代理类,而动态代理则可以在运行时动态生成代理类,更加灵活方便。

本文将介绍Java中的动态代理,包括动态代理的实现原理、使用方法、以及动态代理实现中的常见问题和注意事项。

摘要

本文将分以下几部分介绍Java中的动态代理:

  1. 动态代理的概念和实现原理。
  2. Java中的动态代理实现。
  3. 动态代理的常见问题和注意事项。

内容

1. 动态代理的概念和实现原理

代理模式是一种常用的设计模式,它可以在不改变原有代码的前提下,实现对原有代码的扩展。代理模式主要有静态代理和动态代理两种。

在Java中,静态代理需要手动编写一个代理类,其中代理类需要实现与被代理对象相同的接口方法,并在方法中调用被代理对象的对应方法。而动态代理则是在运行时动态生成代理类,更加灵活方便。

Java中的动态代理主要使用了Java反射机制,通过反射动态生成代理类并调用其中的方法。动态代理需要实现一个代理接口,在运行时使用反射动态生成一个代理类,该代理类实现了代理接口,并在其中调用了InvocationHandler中的invoke()方法。

在Java中,动态代理主要由两个类来实现:java.lang.reflect.Proxy和java.lang.reflect.InvocationHandler。其中,Proxy类用于生成动态代理类对象,而InvocationHandler接口则用于实现具体的代理逻辑。在使用动态代理时,需要创建一个InvocationHandler对象,并将其作为参数传递给Proxy.newProxyInstance()方法,该方法将返回一个代理对象。

2. Java中的动态代理实现

Java中的动态代理主要使用了java.lang.reflect.Proxy和java.lang.reflect.InvocationHandler两个类。在使用动态代理时,需要创建一个InvocationHandler对象,并将其作为参数传递给Proxy.newProxyInstance()方法,该方法将返回一个代理对象。

下面是一个简单的示例,展示了如何使用动态代理:

interface Subject {
    void request();
}

class RealSubject implements Subject {
    public void request() {
        System.out.println("RealSubject request()");
    }
}

class DynamicProxy implements InvocationHandler {
    private Object subject;

    public DynamicProxy(Object subject) {
        this.subject = subject;
    }

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("before");
        Object result = method.invoke(subject, args);
        System.out.println("after");
        return result;
    }
}

public class Test {
    public static void main(String[] args) {
        RealSubject realSubject = new RealSubject();
        InvocationHandler handler = new DynamicProxy(realSubject);
        Subject subject = (Subject) Proxy.newProxyInstance(handler.getClass().getClassLoader(),
                realSubject.getClass().getInterfaces(), handler);
        subject.request();
    }
}

在这个例子中,我们定义了一个Subject接口和一个RealSubject类。DynamicProxy类是我们定义的代理类,用于实现代理逻辑。在实现代理逻辑时,我们使用了InvocationHandler接口,并在其中实现了before()和after()方法,用于在代理的方法执行前后进行处理。

在main()方法中,我们首先创建了一个RealSubject对象,然后创建了一个DynamicProxy对象,将RealSubject对象传递给它。接着,我们使用Proxy.newProxyInstance()方法创建了一个代理对象,并将该代理对象强制转换为Subject类型。最后,我们调用了代理对象的request()方法,该方法会自动调用DynamicProxy中的invoke()方法,从而实现了代理逻辑。

3. 动态代理的常见问题和注意事项

动态代理虽然使用灵活方便,但在实现时也需要注意一些问题:

  1. 基于接口的代理:在Java中,动态代理只能基于接口实现。如果需要基于实现类实现动态代理,则需要使用字节码工具类,例如ASM和CGLIB。
  2. 方法调用循环问题:在动态代理中,如果代理对象调用了被代理对象的方法,将会导致invoke()方法被重复调用,从而导致死循环。为了避免这个问题,我们可以在invoke()方法中判断是否为代理对象,以避免调用被代理对象的方法。
  3. hashCode和equals方法的问题:在动态代理中,由于代理类和被代理类是两个不同的类,因此它们的hashCode和equals方法会有不同的实现。如果需要在代理对象中使用hashCode和equals方法,需要特别处理。

测试用例

interface Calculator {
    int add(int a, int b);
    int subtract(int a, int b);
}

class CalculatorImpl implements Calculator {
    public int add(int a, int b) {
        return a + b;
    }
    public int subtract(int a, int b) {
        return a - b;
    }
}

class CalculatorProxy implements InvocationHandler {
    private Object target;

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

    public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
        System.out.println("before");
        Object result = method.invoke(target, args);
        System.out.println("after");
        return result;
    }
}

public class Test {
    public static void main(String[] args) {
        Calculator calculator = new CalculatorImpl();
        InvocationHandler handler = new CalculatorProxy(calculator);
        Calculator proxy = (Calculator) Proxy.newProxyInstance(handler.getClass().getClassLoader(),
                calculator.getClass().getInterfaces(), handler);
        System.out.println("add(1, 2) = " + proxy.add(1, 2));
        System.out.println("subtract(5, 3) = " + proxy.subtract(5, 3));
    }
}

该测试用例演示了如何使用动态代理实现一个简单的计算器。其中,Calculator接口定义了add()和subtract()两个方法,CalculatorImpl类是真正的计算器实现类,而CalculatorProxy类是代理类,用于实现代理逻辑。

在测试用例中,我们首先创建了一个CalculatorImpl对象,然后创建了一个CalculatorProxy对象,将CalculatorImpl对象传递给它。接着,我们使用Proxy.newProxyInstance()方法创建了一个代理对象,并将该代理对象强制转换为Calculator类型。最后,我们调用了代理对象的add()和subtract()方法,并输出了它们的返回值。

全文小结

本文介绍了Java中的动态代理,包括动态代理的概念和实现原理、Java中的动态代理实现、以及动态代理实现中的常见问题和注意事项。动态代理可以在不改变原有代码的前提下,实现对原有代码的扩展,非常灵活方便。在使用动态代理时,需要注意基于接口的代理、方法调用循环问题以及hashCode和equals方法的问题。

附录源码

  如上涉及所有源码均已上传同步在Gitee,提供给同学们一对一参考学习,辅助你更迅速的掌握。

☀️建议/推荐你


  无论你是计算机专业的学生,还是对编程有兴趣的小伙伴,都建议直接毫无顾忌的学习此专栏「滚雪球学Java」,bug菌郑重承诺,凡是学习此专栏的同学,均能获取到所需的知识和技能,全网最快速入门Java编程,就像滚雪球一样,越滚越大,指数级提升。

📣关于我


我是bug菌,CSDN | 掘金 | infoQ | 51CTO 等社区博客专家,历届博客之星Top30,掘金年度人气作者Top40,51CTO年度博主Top12,华为云 | 阿里云| 腾讯云等社区优质创作者,全网粉丝合计15w+ ;硬核微信公众号「猿圈奇妙屋」,欢迎你的加入!免费白嫖最新BAT互联网公司面试题、4000G pdf电子书籍、简历模板等海量资料。

【版权声明】本文为华为云社区用户原创内容,转载时必须标注文章的来源(华为云社区)、文章链接、文章作者等基本信息, 否则作者和本社区有权追究责任。如果您发现本社区中有涉嫌抄袭的内容,欢迎发送邮件进行举报,并提供相关证据,一经查实,本社区将立刻删除涉嫌侵权内容,举报邮箱: cloudbbs@huaweicloud.com
  • 点赞
  • 收藏
  • 关注作者

评论(0

0/1000
抱歉,系统识别当前为高风险访问,暂不支持该操作

全部回复

上滑加载中

设置昵称

在此一键设置昵称,即可参与社区互动!

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。

*长度不超过10个汉字或20个英文字符,设置后3个月内不可修改。