# 代理模式

# 代理模式

# 定义

由于某些原因需要给某对象提供一个代理以控制对该对象的访问。这时,访问对象不适合或者不能直接引用目标对象,代理对象作为访问对象和目标对象之间的中介。

# 结构

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

# UML类图

image-20211111205014802

举个栗子

比如女朋友经常找人代购化妆品,代购人则是代理对象,你想买房,中介就是代理对象,你想买车,经销商就是代理对象。

而 JDK 对于代理模式的实现一般有三种:静态代理、动态代理、cglib动态代理。

# 静态代理

假如我们需要买一辆奔驰,那就会去找奔驰的经销商。

Car 接口

image-20211111192639900

实现类

image-20211111192708710

代理类

image-20211111192724670

测试

image-20211111192804987

如果说我们需要给汽车买一些配件,而上面的经销商只卖车,并没有提供其他的服务,因此我们最好去 4s 店购买。

缺点:对于已经定义好的代理类,如果新增功能,会违背开闭原则,并且增加维护成本。对于这种情况,可以通过动态代理的方式去解决。

# JDK动态代理

在JDK中的动态代理提供了一个类库,可以直接使用,不依赖第三方。

动态代理是指,程序在整个运行过程中根本就不存在目标类的代理类,目标对象的代理 对象只是由代理生成工具(不是真实定义的类)在程序运行时由 JVM 根据反射等机制动态生成的。代理对象与目标对象的代理关系在程序运行时才确立。

对比静态代理,静态代理是指在程序运行前就已经定义好了目标类的代理类。代理类与目标类的代理关系在程序运行之前就确立了。

动态代理的实现方式常用的有两种:使用 JDK 的 Proxy,与通过 CGLIB 生成代理。

对于JDK动态代理,先看代码:

在原 Car 接口添加方法

image-20211111204425924

实现类

image-20211111204340424

动态代理类

image-20211111205305877

测试

image-20211111202523442

JDK内置的Proxy动态代理可以在运行时动态生成字节码,而没必要针对每个类编写代理类,并且不需要实现业务接口(Car)。中间主要使用到了一个接口InvocationHandlerProxy.newProxyInstance静态方法。

缺点:JDK动态代理的代理对象在创建时,需要有业务实现类所实现的接口作为参数(因为后面代理方法需要根据接口内的方法名进行调用)。如果业务实现类没有实现接口而是直接定义接口的话,或者该业务实现类中增加了接口没有的方法(因为无法调用),就无法使用JDK动态代理。

于是可以通过CGLIB 动态代理解决这个问题。

# CGLIB动态代理

CGLIB(Code Generation Library)是一个开源项目,是一个强大的,高性能,高质量的Code生成类库,它可以在运行期扩展Java类与实现 Java 接口,通俗说 CGLIB 可以在运行时动态生成字节码。

JDK 的 Proxy是基于接口动态代理 ,CGLIB 则是基于类的动态代理。

引入 maven 依赖

<!-- https://mvnrepository.com/artifact/cglib/cglib -->
<dependency>
    <groupId>cglib</groupId>
    <artifactId>cglib</artifactId>
    <version>3.3.0</version>
</dependency>
1
2
3
4
5
6

目标类

image-20211112112653212

cglib 代理类

image-20211112112737925

测试

image-20211112112806451

其中 Enhancer 对象把代理对象设置为被代理类的子类来实现动态代理,并且能代理普通类。

对于 CGLIB 详细介绍参考:CGLIB详解 (opens new window)

# 三者的区别

image-20211112113913820