热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

理解Java动态代理需要对Java的反射机制有一定了解什么是代理模式在有些情况下,一个客户不能或者不想直接访问另一个对象,这时需要找一个中介帮忙完成某

理解Java动态代理需要对Java的反射机制有一定了解

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节


什么是代理模式

在有些情况下,一个客户不能或者不想直接访问另一个对象,这时需要找一个中介帮忙完成某项任务,这个中介就是代理对象。

例如,购买火车票不一定要去火车站买,可以通过 12306 网站或者去火车票代售点买票。又如找女朋友、找保姆、找工作等都可以通过找中介完成。

定义

由于某些原因需要给某对象提供一个代理以控制对该对象的访问。

访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。

代理模式的主要角色
  • 抽象角色(Subject):通过接口或抽象类声明真实主题和代理对象实现的业务方法。
  • 真实角色(Real Subject):实现了抽象主题中的具体业务,是代理对象所代表的真实对象,是最终要引用的对象。
  • 代理(Proxy):提供了与真实主题相同的接口,其内部含有对真实主题的引用,它可以访问、控制或扩展真实主题的功能。
  • 客户:使用代理角色来进行一些操作。

优点

代理模式在客户端与目标对象之间起到一个中介作用和保护目标对象的作用

代理对象可以扩展目标对象的功能

代理模式能将客户端与目标对象分离,在一定程度上降低了系统的耦合度,增加了程序的可扩展性

缺点

冗余,由于代理对象要实现与目标对象一致的接口,会产生过多的代理类。

系统设计种类的数量增加,变得难以维护。

使用动态代理方式,可以有效避免以上的缺点

静态代理

静态代理其实就是最基础、最标准的代理模式实现方案。

举例:

Rent . java 即抽象角色

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

Landlord . java 即真实角色

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

Proxy . java 即代理

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

Client . java 即客户

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

结果:

  • 带房客看房
  • 房屋出租
  • 收中介费
  • Process finished with exit code 0

在这个过程中,客户接触的是中介,看不到房东,但是依旧租到了房东的房子。同时房东省了心,客户省了事。

静态代理享受代理模式的优点,同时也具有代理模式的缺点,那就是一旦实现的功能增加,将会变得异常冗余和复杂,秒变光头。

为了保护头发,就出现了动态代理模式!

动态代理

动态代理的出现就是为了解决传统静态代理模式的中的缺点。

具备代理模式的优点的同时,巧妙地解决了静态代理代码冗余,难以维护的缺点。

在Java中常用的有如下几种方式:

  • JDK 原生动态代理
  • cglib 动态代理
  • javasist 动态代理

JDK原生动态代理

上例中静态代理类中,中介作为房东的代理,实现了相同的租房接口。

例子

首先实现一个InvocationHandler,方法调用会被转发到该类的invoke()方法。

然后在需要使用Rent的时候,通过JDK动态代理获取Rent的代理对象。

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

客户使用动态代理调用

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

运行结果和前例相同

上述代码的核心关键是Proxy.newProxyInstance方法,该方法会根据指定的参数动态创建代理对象。

它三个参数的意义如下:

  1. loader,指定代理对象的类加载器
  2. interfaces,代理对象需要实现的接口,可以同时指定多个接口
  3. handler,方法调用的实际处理者,代理对象的方法调用都会转发到这里

Proxy.newProxyInstance会返回一个实现了指定接口的代理对象,对该对象的所有方法调用都会转发给InvocationHandler.invoke()方法。

因此,在invoke()方法里我们可以加入任何逻辑,比如修改方法参数,加入日志功能、安全检查功能等等等等……

小结

显而易见,对于静态代理而言,我们需要手动编写代码让代理实现抽象角色的接口。

而在动态代理中,我们可以让程序在运行的时候自动在内存中创建一个实现抽象角色接口的代理,而不需要去单独定义这个类,代理对象是在程序运行时产生的,而不是编译期。

对于从Object中继承的方法,JDK Proxy会把hashCode()、equals()、toString()这三个非接口方法转发给InvocationHandler,其余的Object方法则不会转发。

CGLIB动态代理

JDK动态代理是基于接口的,如果对象没有实现接口该如何代理呢?CGLIB代理登场

CGLIB(Code Generation Library)是一个基于ASM的字节码生成库,它允许我们在运行时对字节码进行修改和动态生成。CGLIB通过继承方式实现代理。

使用cglib需要引入cglib的jar包,如果你已经有spring-core的jar包,则无需引入,因为spring中包含了cglib。

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

来看示例,假设我们有一个没有实现任何接口的类Landlord:

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

因为没有实现接口,所以使用通过CGLIB代理实现如下:

首先实现一个MethodInterceptor,方法调用会被转发到该类的intercept()方法

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

客户通过CGLIB动态代理获取代理对象

从源码的角度搞懂Java代理模式,那些面试中你最容易忽略的细节

运行输出结果和前例相同

对于从Object中继承的方法,CGLIB代理也会进行代理,如hashCode()、equals()、toString()等,但是getClass()、wait()等方法不会,因为它是final方法,CGLIB无法代理。

其实CGLIB和JDK代理的思路大致相同

上述代码中,通过CGLIB的Enhancer来指定要代理的目标对象、实际处理代理逻辑的对象。

最终通过调用create()方法得到代理对象,对这个对象所有非final方法的调用都会转发给
MethodInterceptor.intercept()方法。

在intercept()方法里我们可以加入任何逻辑,同JDK代理中的invoke()方法

通过调用MethodProxy.invokeSuper()方法,我们将调用转发给原始对象,具体到本例,就是Landlord的具体方法。CGLIG中MethodInterceptor的作用跟JDK代理中的InvocationHandler很类似,都是方法调用的中转站。

final类型

CGLIB是通过继承的方式来实现动态代理的,有继承就不得不考虑final的问题。我们知道final类型不能有子类,所以CGLIB不能代理final类型,遇到这种情况会抛出类似如下异常:

java.lang.IllegalArgumentException: Cannot subclass final class cglib.HelloConcrete

同样的,final方法是不能重载的,所以也不能通过CGLIB代理,遇到这种情况不会抛异常,而是会跳过final方法只代理其他方法。

其他方案

  • 使用ASM在被代理类基础上生成新的字节码形成代理类
  • 使用javassist在被代理类基础上生成新的字节码形成代理类

javassist也是常用的一种动态代理方案,ASM速度非常快,这里不在进行展开。

尾声

动态代理是[Spring AOP(https://jq.qq.com/?_wv=1027&k=0IsBuUb0)(Aspect Orient Programming, 面向切面编程)的实现方式,了解动态代理原理,对理解Spring AOP大有帮助。

  • 如spring等这样的框架,要增强具体业务的逻辑方法,不可能在框架里面去写一个静态代理类,太蠢了,只能按照用户的注解或者xml配置来动态生成代理类。
  • 业务代码内,当需要增强的业务逻辑非常通用(如:添加log,重试,统一权限判断等)时,使用动态代理将会非常简单,如果每个方法增强逻辑不同,那么静态代理更加适合。
  • 使用静态代理时,如果代理类和被代理类同时实现了一个接口,当接口方法有变动时,代理类也必须同时修改,代码将变得臃肿且难以维护。

推荐阅读
  • 本文介绍了Hyperledger Fabric外部链码构建与运行的相关知识,包括在Hyperledger Fabric 2.0版本之前链码构建和运行的困难性,外部构建模式的实现原理以及外部构建和运行API的使用方法。通过本文的介绍,读者可以了解到如何利用外部构建和运行的方式来实现链码的构建和运行,并且不再受限于特定的语言和部署环境。 ... [详细]
  • Java程序设计第4周学习总结及注释应用的开发笔记
    本文由编程笔记#小编为大家整理,主要介绍了201521123087《Java程序设计》第4周学习总结相关的知识,包括注释的应用和使用类的注释与方法的注释进行注释的方法,并在Eclipse中查看。摘要内容大约为150字,提供了一定的参考价值。 ... [详细]
  • 提升Python编程效率的十点建议
    本文介绍了提升Python编程效率的十点建议,包括不使用分号、选择合适的代码编辑器、遵循Python代码规范等。这些建议可以帮助开发者节省时间,提高编程效率。同时,还提供了相关参考链接供读者深入学习。 ... [详细]
  • 本文介绍了lua语言中闭包的特性及其在模式匹配、日期处理、编译和模块化等方面的应用。lua中的闭包是严格遵循词法定界的第一类值,函数可以作为变量自由传递,也可以作为参数传递给其他函数。这些特性使得lua语言具有极大的灵活性,为程序开发带来了便利。 ... [详细]
  • 本文介绍了数据库的存储结构及其重要性,强调了关系数据库范例中将逻辑存储与物理存储分开的必要性。通过逻辑结构和物理结构的分离,可以实现对物理存储的重新组织和数据库的迁移,而应用程序不会察觉到任何更改。文章还展示了Oracle数据库的逻辑结构和物理结构,并介绍了表空间的概念和作用。 ... [详细]
  • 开发笔记:加密&json&StringIO模块&BytesIO模块
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了加密&json&StringIO模块&BytesIO模块相关的知识,希望对你有一定的参考价值。一、加密加密 ... [详细]
  • Android中高级面试必知必会,积累总结
    本文介绍了Android中高级面试的必知必会内容,并总结了相关经验。文章指出,如今的Android市场对开发人员的要求更高,需要更专业的人才。同时,文章还给出了针对Android岗位的职责和要求,并提供了简历突出的建议。 ... [详细]
  • 如何用UE4制作2D游戏文档——计算篇
    篇首语:本文由编程笔记#小编为大家整理,主要介绍了如何用UE4制作2D游戏文档——计算篇相关的知识,希望对你有一定的参考价值。 ... [详细]
  • XML介绍与使用的概述及标签规则
    本文介绍了XML的基本概念和用途,包括XML的可扩展性和标签的自定义特性。同时还详细解释了XML标签的规则,包括标签的尖括号和合法标识符的组成,标签必须成对出现的原则以及特殊标签的使用方法。通过本文的阅读,读者可以对XML的基本知识有一个全面的了解。 ... [详细]
  • 本文介绍了在Windows环境下如何配置php+apache环境,包括下载php7和apache2.4、安装vc2015运行时环境、启动php7和apache2.4等步骤。希望对需要搭建php7环境的读者有一定的参考价值。摘要长度为169字。 ... [详细]
  • Java在运行已编译完成的类时,是通过java虚拟机来装载和执行的,java虚拟机通过操作系统命令JAVA_HOMEbinjava–option来启 ... [详细]
  • 本文介绍了Python爬虫技术基础篇面向对象高级编程(中)中的多重继承概念。通过继承,子类可以扩展父类的功能。文章以动物类层次的设计为例,讨论了按照不同分类方式设计类层次的复杂性和多重继承的优势。最后给出了哺乳动物和鸟类的设计示例,以及能跑、能飞、宠物类和非宠物类的增加对类数量的影响。 ... [详细]
  • GreenDAO快速入门
    前言之前在自己做项目的时候,用到了GreenDAO数据库,其实对于数据库辅助工具库从OrmLite,到litePal再到GreenDAO,总是在不停的切换,但是没有真正去了解他们的 ... [详细]
  • 本文整理了Java面试中常见的问题及相关概念的解析,包括HashMap中为什么重写equals还要重写hashcode、map的分类和常见情况、final关键字的用法、Synchronized和lock的区别、volatile的介绍、Syncronized锁的作用、构造函数和构造函数重载的概念、方法覆盖和方法重载的区别、反射获取和设置对象私有字段的值的方法、通过反射创建对象的方式以及内部类的详解。 ... [详细]
  • Spring框架《一》简介
    Spring框架《一》1.Spring概述1.1简介1.2Spring模板二、IOC容器和Bean1.IOC和DI简介2.三种通过类型获取bean3.给bean的属性赋值3.1依赖 ... [详细]
author-avatar
染伊沐2010
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有