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


🏆本文收录于「滚雪球学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电子书籍、简历模板等海量资料。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.rhkb.cn/news/137370.html

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

IPv6协议基本概念

目前大多数设备节点支持IPv6和IPv4双栈,但随着IPv6逐渐应用,某些设备已经只支持纯IPv6,即IPv6 Only。 一、IPv6地址格式 1、IPv6地址表示方式 IPv6可以写成一组8个十六进制数,用冒号(:)分割。也可以写成128位2进制的0和1。 32即8x4,8表示8组16进制数,4表示每组16禁止包…

Git基础操作

前言 本文会向您介绍如何安装git,以及快速地上手add,commit,push,版本回退操作 基础配置 关于windous上的安装git官网已经介绍的很清楚了,您可以直接点入链接windows安装 如果你的平台是centos,以centos…

【Verilog教程】2.4Verilog表达式

表达式 表达式由操作符和操作数构成,其目的是根据操作符的意义得到一个计算结果。表达式可以在出现数值的任何地方使用。 例如: a^b ; //a与b进行异或操作 address[9:0] 10b1 ; //地址累加 flag1 && flag2 ; //逻辑与操作操作数 …

有没有普通人可以做的项目?分享几个项目适合普通人做的!

当谈到副业时,我们通常会想到能够轻松实施的兼职方式,并且希望通过这些副业增加我们的收入。那么,以下是我推荐的几种副业,这些副业不仅有良好的收入潜力,而且在执行过程中也很有趣。让我们一起看看吧! 第…

Hive行转列[一行拆分成多行/一列拆分成多列]

场景: hive有张表armmttxn_tmp,其中有一个字段lot_number,该字段以逗号分隔开多个值,每个值又以冒号来分割料号和数量,如:A3220089:-40,A3220090:-40,A3220091:-40,A3220083:-40,A3220087:-40,A3220086:-4…

爬虫框架Scrapy学习笔记-2

前言 Scrapy是一个功能强大的Python爬虫框架,它被广泛用于抓取和处理互联网上的数据。本文将介绍Scrapy框架的架构概览、工作流程、安装步骤以及一个示例爬虫的详细说明,旨在帮助初学者了解如何使用Scrapy来构建和运行自己的网络爬虫。 爬虫框架Scrapy学…

Redis的介绍以及简单使用

Redis(Remote Dictionary Server)是一个开源的内存数据存储系统,它以键值对的形式将数据存在内存中,并提供灵活、高性能的数据访问方式。Redis具有高速读写能力和丰富的数据结构支持,可以广泛应用于缓存、消息队列、实…

华为云云耀云服务器L实例评测|轻量级应用服务器对决:基于 STREAM 深度测评华为云云耀云服务器L实例的内存性能

本文收录在专栏:#云计算入门与实践 - 华为云 专栏中,本系列博文还在更新中 相关华为云云耀云服务器L实例评测文章列表如下: 华为云云耀云服务器L实例评测 | 从零开始:云耀云服务器L实例的全面使用解析指南华为云云耀云服务器L实…

Outlook打开超链接用默认浏览器Microsoft outlook open hyperlink using default browser

这两天outlook打开超链接一直用edge,但我的默认浏览器是chrome。 解决方法 在outlook的选项中的高级设置里面,将超链接打开选为默认浏览器。

ESP8266 Windows下开发环境搭建

文章目录 1、SDK下载2、Windows下的开发工具链安装3、ESP8266专用工具链安装4、运行msys32目录下的mingw32.exe,初始化环境5、配置环境变量6、编译hello world例程7、SDK配置8、烧录9、在串口助手中查看 1、SDK下载 官方提供了两种SDK,OS版本和非OS版本…

java springboot 如何实现小程序支付

今天给大家分享java小程序支付 首先我们学习任何东西要先看官网 下面是支付业务流程 我们具体用代码去实现上面的业务流程 功能截图 代码截图 pay(){//调用后台生成订单var orderNumber "20210101123456";var amount 0.01;WxPay.wxpay(app, amount, orderNumber…

(一)探索随机变量及其分布:概率世界的魔法

文章目录 🍋引言🍋什么是随机变量?🍋离散随机变量🍋连续随机变量 🍋随机变量的概率分布🍋离散概率分布🍋0-1分布(Bernoulli分布)🍋二项分布&#…

Unity制作曲线进度条

unity制作曲线进度条 大家好,我是阿赵。   在使用Unity引擎做进度条的时候,有时会遇到一个问题,如果进度条不是简单的横向、纵向或者圆形,而是任意的不规则形状,那该怎么办呢?比如这样的: 一…

CentOS7安装Nginx+ModSecurity

一、介绍 当学习网络安全时,了解和使用安全设备是必不可少的一部分,其中一种常见的安全设备是Web应用防火墙(WAF)。市场上有许多商业化的WAF,但对于学习目的,我推荐使用一款免费开源的WAF,名为…

SpringMVC自定义注解和使用

一.引言 1.简介: 在Spring MVC中,我们可以使用自定义注解来扩展和定制化我们的应用程序。自定义注解是一种通过Java的注解机制定义的特殊注解,可以应用于控制器类、方法或者方法参数上,以实现不同的功能和行为。(注解…

【小记录】jupyter notebook新版本

手欠升级 😅今天手贱,在anaconda navigator里面更新了最新版本的spyder,然后莫名奇妙地jupyter notebook就打不开了😅,报错说缺少模块”ModuleNotFoundError: No module named jupyter_server.contents“,…

4G版本云音响设置教程腾讯云平台版本

文章目录 4G本云音响设置教程介绍一、申请设备三元素1.腾讯云物联网平台2.创建产品3.设置产品参数4.添加设备5.获取三元素 二、设置设备三元素1.打开MQTTConfigTools2.计算MQTT参数3.使用USB连接设备4.设置参数 三、腾讯云物联网套件协议使用说明1.推送协议信息2.topic规则说明…

解决连接数据库提示:Public Key Retrieval is not allowed

最近在使用新的用户连接mysql时,总是提示:Public Key Retrieval is not allowed 解决方法一:在(连接属性)添加allowPublicKeyRetrievaltrue 解决方法二(不建议):先在cmd上登录

分类预测 | Matlab实现RBF-Adaboost多特征分类预测

分类预测 | Matlab实现RBF-Adaboost多特征分类预测 目录 分类预测 | Matlab实现RBF-Adaboost多特征分类预测效果一览基本介绍研究内容程序设计参考资料 效果一览 基本介绍 1.Matlab实现基于RBF-Adaboost数据分类预测(Matlab完整程序和数据) 2.多特征输入…

编译工具:CMake(八) | cmake 常用指令

编译工具:CMake(八) | cmake 常用指令 基本指令 基本指令 ADD_DEFINITIONS向 C/C编译器添加-D 定义,比如:ADD_DEFINITIONS(-DENABLE_DEBUG-DABC),参数之间用空格分割。 如果你的代码中定义了#ifdef ENABLE_DEBUG #end…