Java高级编程细节-动态代理-进阶高级开发必学技能
关于代理模式的话题有很多,
在开发中经常用到的应该是静态代理模式,能很好的去耦合。
动态代理是代理模式的另外一种实现。
动态代理的区别在哪里?
动态代理有什么好处?
今天我们来分析下这些问题。
回顾静态代理
之前我们分析过一次静态代理,
用代理模式优雅地写代码
一个典型的代理模式的 Proxy类像下面这样,
对于调用者来说,需要把构造好的实例传给代理,然后就可以用代理来替代操作真正的实例了。
静态代理的问题是,
在接口代码少的情况下一切没什么问题,但是当接口增加的时候,
Proxy 类就需要响应的增加接口,比方上面的 Func 接口,
刚开始可能只有一个 read()方法,后面慢慢发展到有了 write(),有了 mark(),
随着接口量的增加, Proxy的维护工作量也在逐步增加。
那么动态代理能怎么解决这种问题呢?
动态代理的实现
动态代理的实现步骤基本如下:
· 定义一个公共接口(像 Func)和实现类(像 User),这部分跟静态代理一样
· 定义一个 DynamicProxy类实现 InvocationHandler 接口,这个Proxy类似于静态代理的 Proxy然而不用实现 Func接口
· 调用 Proxy类来构造代理对象
在动态代理的实现中有两个东西非常重要,一个是 Proxy类,一个是 InvocationHandler接口,都位于 reflect包下,
我们来看第二个步骤所定义的 DynamicProxy是怎样的吧
它同样持有了委托对象实例,但是和静态代理不同,
它并没有实现委托对象的接口方法,
而只实现了 InvocationHandler的 invoke方法,然后调用了 method.invoke(this.user, objects);
可以留意下 invoke方法,后面我们继续分析,
这里再贴一下 Client类的代码,也就是使用 Proxy的地方,
跟静态代理不同的地方在于,虽然这里也需要实例化一个委托类的对象,并传给 Proxy的构造方法,
但这里所实例化的是 InvocationHandler对象,而不是 DynamicProxy的对象。
到这里就完成了一个动态代理的代码,输出结果如下
before invoke user method
user read
after invoke user method
比较&分析
· 先来说第二个步骤的 DynamicProxy的实现,
可以发现跟静态代理不同的地方在于,静态代理需要实现 Func接口的 read()方法,而动态代理实现的是 InvocationHandler的 invoke方法,
静态代理需要在不同的接口中去调用 User 接口的不同方法,
而动态代理在invoke被调用的过程中不需要关心需要调用 User 的哪个具体方法,
方法被封装在 method对象中,而所需要的参数则在 Object[] objects,
直接调用就可以
这里就意味着即使以后增加了 Func的接口,对于 DynamicProxy来说也不需要增加额外的维护量。
· Proxy.newProxyInstance干了什么
在静态代理里面,我们会直接用 new StaticProxy(user)构造出来的静态对象直接操作,
而在动态代理里面,我们操作的是 Proxy.newProxyInstance所构造出来的动态代理对象 proxyUser,
可能初次接触动态代理的同学在这里就概念混乱了,
"难道代理类不是 new DynamicProxy出来的对象吗?"
其实不是的,如果把 proxyUser的类名打印出来的话,
它会以 $ProxyN的形式存在,N从0开始,这个就是动态代理所生成的真正代理对象,
动态代理的意义就在于这里,$ProxyN 这个对象是在运行时创建的,
如果用代码来解释的话,$ProxyN的代码会像下面这样
这个才是真正的代理类。
欢迎工作一到五年的Java工程师朋友们加入Java架构开发:468947140
点击链接加入群聊【Java-BATJ企业级资深架构】:https://jq.qq.com/?_wv=1027&k=5zMN6JB
本群提供免费的学习指导 架构资料 以及免费的解答
不懂得问题都可以在本群提出来 之后还会有职业生涯规划以及面试指导
对本文感兴趣欢迎关注笔者,如有不同意见欢迎留言。
低调大师中文资讯倾力打造互联网数据资讯、行业资源、电子商务、移动互联网、网络营销平台。
持续更新报道IT业界、互联网、市场资讯、驱动更新,是最及时权威的产业资讯及硬件资讯报道平台。
转载内容版权归作者及来源网站所有,本站原创内容转载请注明来源。
- 上一篇
到底如何高效学习?
IT 行业是一个变化非常快的行业,它需要我们持续去学习新的知识和技能。 但是,工作以后,我们经常会发现自己学习的东西很少了,倒不是没有时间去学习, 而是学习的效率太低了。久而久之,就演变成『一年的工作经验,重复用十年』。 当然,有些人会说自己经常加班,没有时间学习,这只是表象,时间挤挤总是有的。 你想想你为了上王者,浪费了多少时间?为了刷今日头条,又消磨了多少光阴? 另外,很多人推崇碎片化学习,但是有一些东西碎片化学习效率是很低的,比如数学。 1一些学习的坏习惯 1.1被动反复阅读 通常编程新手在学习一个新东西的时候,喜欢买一本权威指南之类的书(大神或者同事推荐),比如「C++ Primer」和「Javascript 权威指南」。 而这样一本书,一般页数在700-1400页左右,要完整读完,在不求甚解的基础之上大概要花费好几个月甚至大半年时间。 别说是新手,就算是一个C++编程老手去读「Javascript 权威指南」这样的书也不可能在只阅读一遍之后就能理解。 这时,很多人会选择重复多次阅读。有人会从头开始重复阅读,也有人只挑不理解的章节来阅读。 我以前上大学那会儿就是这么干的,读了好...
- 下一篇
Python全栈工程师 (exercises)
# 1:给定一个数,判断他是正数,负数,还是0 a = int(input("请输入一该个整数")) if a == 0: print(a, "是0") elif a > 0: print(a, "是正数") else: print(a, "是负数") # 练习2:输入一个季度的数字 输出该季度有那几个月 b = int(input("请输入一个季度:1.春季2.夏季3.秋季4.冬季")) if b > 4: print(b, "不是季节") elif b < 0: print(b, "不是季节") else: if b == 1: print(b, "春季有:一月,二月,三月") elif b == 2: print(b, "夏季有:四月,五月,六月") elif b == 3: print(b, "秋季有:七月,八月,九月") else: print(b, "冬季有:十月,十一月,十二月") # 联系3:输入一个月份 输出这个月份在那个季度 c = int(input("请输入一个月份判断在那个季度")) if c < 0: print(c, "不是月份")...
相关文章
文章评论
共有0条评论来说两句吧...
文章二维码
点击排行
推荐阅读
最新文章
- CentOS8编译安装MySQL8.0.19
- CentOS8,CentOS7,CentOS6编译安装Redis5.0.7
- SpringBoot2整合MyBatis,连接MySql数据库做增删改查操作
- SpringBoot2整合Redis,开启缓存,提高访问速度
- SpringBoot2配置默认Tomcat设置,开启更多高级功能
- Hadoop3单机部署,实现最简伪集群
- CentOS7,CentOS8安装Elasticsearch6.8.6
- CentOS6,7,8上安装Nginx,支持https2.0的开启
- Docker使用Oracle官方镜像安装(12C,18C,19C)
- SpringBoot2编写第一个Controller,响应你的http请求并返回结果