【Java设计模式】六、代理模式:静态代理、JDK + CGLIB动态代理

文章目录

  • 1、代理对象
  • 2、代理模式结构
  • 3、静态代理
  • 4、JDK动态代理
  • 5、JDK动态代理的原理
  • 6、CGLIB动态代理
  • 7、三种代理的对比
  • 8、代理模式的总结

结构型设计是将类或者对象按某种布局(继承机制、组合聚合)来组成更大结构。包括七种:

* 代理模式
* 适配器模式
* 装饰者模式
* 桥接模式
* 外观模式
* 组合模式
* 享元模式

1、代理对象

以买电脑为例,联想公司就是目标对象,地方代理商就是代理对象。

在这里插入图片描述

Java按照代理对象生成时机,分为:

  • 静态代理:代理类在编译器就生成
  • 动态代理:代理类在程序运行时动态生成。又分为JDK代理和CGLib代理

2、代理模式结构

有三种角色:

  • 抽象主题类:定义一套规范
  • 真实主题类:下面例子中的火车站类
  • 代理类:下面例子中的代售点类

3、静态代理

直接去火车站买票,需要经历:坐车到火车站、排队等操作,麻烦(即目标对象不适合或不能直接访问到)。我们一般会到镇上就近的火车票代售点。此时,火车站为目标对象,代售点是代理对象。类图如下:注意,代理对象ProxyPoint聚合了目标对象(火车站TrainStation)

在这里插入图片描述

定义卖火车票的规范:

//卖票接口
public interface SellTickets {void sell();
}

定义火车站,实现规范接口:

//火车站  火车站具有卖票功能,所以需要实现SellTickets接口
public class TrainStation implements SellTickets {public void sell() {System.out.println("火车站卖票");}
}

定义代售点类,和目标对象实现同一个接口(即抽象主题),聚合目标对象,代售点卖票,调用的也是目标对象(火车站对象)的卖票功能:

//代售点
public class ProxyPoint implements SellTickets {private TrainStation station = new TrainStation();    //目标对象public void sell() {System.out.println("代理点收取一些服务费用");station.sell();}
}

客户端测试:

public class Client {public static void main(String[] args) {ProxyPoint proxy = new ProxyPoint();   //创建代理对象proxy.sell();}
}

注意上面代理类自己sell方法里的这句:

System.out.println("代理点收取一些服务费用");

客户端直接访问的是代理对象,代理对象是一个目标对象和访问者的中介,同时也是对原来对象的一个增强!!!(如上面代理对象的sell就新增了收取服务费用的功能)

4、JDK动态代理

先是抽象主题和目标对象:

//卖票接口
public interface SellTickets {void sell();
}//火车站  火车站具有卖票功能,所以需要实现SellTickets接口
public class TrainStation implements SellTickets {public void sell() {System.out.println("火车站卖票");}
}

JDK中有一个类Proxy,它不是上面提到的代理对象类,而是提供创建代理对象的静态方法newProxyInstance来获取代理对象

//代理工厂,用来创建代理对象
public class ProxyFactory {//声明目标对象private TrainStation station = new TrainStation();   //返回代理对象public SellTickets getProxyObject() {//使用Proxy获取代理对象/*newProxyInstance()方法参数说明:ClassLoader loader : 类加载器,用于加载代理类,使用真实对象的类加载器即可Class<?>[] interfaces : 真实对象所实现的接口,但代理模式下,真实对象和代理对象实现相同的接口,这里依旧用目标对象(真实对象)+ getInterfaces拿到接口类对象InvocationHandler h : 代理对象的调用处理程序*/SellTickets proxyObject = (SellTickets) Proxy.newProxyInstance(station.getClass().getClassLoader(),station.getClass().getInterfaces(),new InvocationHandler() {/*InvocationHandler中invoke方法参数说明:proxy : 代理对象,即上面的proxyObjectmethod : 对应于在代理对象上调用的接口方法的 Method 实例args : 通过代理对象调用接口方法时传递的实际参数*/public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("代理点收取一些服务费用(JDK动态代理方式)");//执行真实对象,调用method的invoke,传入调用方法的对象和方法的实参Object result = method.invoke(station, args);return result;}});return proxyObject;}
}

客户端:

//测试类
public class Client {public static void main(String[] args) {//创建代理对象工厂ProxyFactory factory = new ProxyFactory();   //获取代理对象(proxyObject.getClass发现其全类名是com.sun.proxy.$Proxy0)SellTickets proxyObject = factory.getProxyObject();  //通过代理对象调用买票的方法proxyObject.sell();}
}

运行:

代理点收取一些服务费用(JDK动态代理方式)
火车站卖票

5、JDK动态代理的原理

注意上面定义的ProxyFactory不是代理类,代理类是程序运行过程中动态在内存中生成的类。使用阿尔萨斯下的jad指令,查看生成的代理类$Proxy0:

//简略版,删掉了hashcode、equals等方法
com.sun.proxy.$Proxy0
//程序运行过程中动态生成的代理类public final class $Proxy0 extends Proxy implements SellTickets {private static Method m3;public $Proxy0(InvocationHandler invocationHandler) {super(invocationHandler);}static {m3 = Class.forName("com.myplat.proxy.dynamic.jdk.SellTickets").getMethod("sell", new Class[0]);}public final void sell() {this.h.invoke(this, m3, null);}}

而其父类Proxy:

//Java提供的动态代理相关类public class Proxy implements java.io.Serializable {protected InvocationHandler h;protected Proxy(InvocationHandler h) {this.h = h;}}.....

发现:

  • 代理类$Proxy0实现了抽象主题SellTickets接口(符合:真实的目标对象和代理对象实现了同样的接口)
  • 代理类$Proxy0又将我在newProxyInstance传入的InvocationHandler传给了父类的h属性

因此,以上的完整过程是:

  • 测试类中通过代理对象调用sell方法
  • 到了$Proxy0的sell方法中
  • h.invoke即newProxyInstance方法中传入的第三个匿名内部类对象h中重写的invoke方法
    在这里插入图片描述
  • 传入invoke的三个实参为:proxy代理对象自己(即this),m3(静态代码块中初始化的抽象主题SellTickets的sell方法),sell方法的实参(null)
  • invoke方法中则通过反射,执行真实对象所属类TrainStation中的sell方法并返回,相关代码:Object result = method.invoke(station, args);

6、CGLIB动态代理

前面JDK动态代理Proxy.newProxyInstance方法的第二个形参是真实对象所实现的接口,当上面的案例没有抽象主题接口SellTickets时,则JDK代理无法使用了(看阿尔萨斯拿到的代理类$Proxy0,静态代码块要Class.forName加载newProxyInstance传入的抽象主题接口,现在没这个接口了,自然不能用了)

此时可用CGLIB:高性能的代码生成包

<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>2.2.2</version>
</dependency>

还是买票的案例,这次只有火车站这个具体主题类,没有SellTickets接口:

//火车站
public class TrainStation {public void sell() {System.out.println("火车站卖票");}
}

写代理工厂类,实现MethodInterceptor 接口,重写intercept方法。Enhancer的Callback设置回调,需要一个MethodInterceptor的子实现对象,此时传入代理工厂类对象后,就会回调intercept方法,intercept方法中又调用了真实对象的sell方法。关键点:

  • Enhancer对象
  • 设置父类字节码对象为真实目标对象
  • 设置回调,回调中调真实目标对象的对应方法
//代理工厂
public class ProxyFactory implements MethodInterceptor {//声明真实目标对象private TrainStation  station = new TrainStation();//通过CGLIB获取代理对象public TrainStation getProxyObject() {//创建Enhancer对象,类似于JDK动态代理的Proxy类,下一步就是设置几个参数Enhancer enhancer =new Enhancer();//设置父类的字节码对象enhancer.setSuperclass(TrainStation.class);//设置回调函数enhancer.setCallback(this);//创建代理对象TrainStation proxyObj = (TrainStation) enhancer.create();return proxyObj;}/*intercept方法参数说明:o : 代理对象method : 真实对象中的方法的Method实例args : 实际参数methodProxy :代理对象中的方法的method实例*/public Object intercept(Object o, Method method, Object[] args, MethodProxy methodProxy) throws Throwable {//对原真实目标对象的方法按实际需求做点增强System.out.println("代理点收取一些服务费用(CGLIB动态代理方式)");//调用真实目标对象的方法Object result = method.invoke(station, args);//也可用invokeSuper调用目标对象的方法(因为前面设置了代理对象的父类字节码对象为真实目标对象,他们之间是父子关系)//TrainStation result = (TrainStation) methodProxy.invokeSuper(o, args);return result;}
}

PS:上面的真实目标对象,即目标对象

//测试类
public class Client {public static void main(String[] args) {//创建代理工厂对象ProxyFactory factory = new ProxyFactory();//获取代理对象TrainStation proxyObject = factory.getProxyObject();proxyObject.sell();}
}

7、三种代理的对比

关于JDK代理和CGLIB代理:

  • CGLIB底层用ASM字节码生成框架来生成代理类
  • CGLIB不能对final关键字的类进行代理,因为enhancer.setSuperclass,动态生成的代理类是目标类的子类,而final类无法被继承
  • JDK和CGLIB动态代理的效率,和JDK版本有关(JDK1.8后,JDK > CGLIB)
  • 总之,有接口用JDK动态代理,没接口,走CGLIB

关于动态和静态代理:

当抽象主题接口新增了一个方法,则静态代理下的写的代理类也需要去实现此方法,维护复杂。动态代理无此问题。

8、代理模式的总结

优点:

  • 代理模式在客户端与目标对象之间起到了一个中介作用 + 保护目标对象
  • 代理对象中可以扩展目标对象的功能(上面代售点加服务费、AOP增强)
  • 代理模式将客户端和目标对象解耦

缺点:

  • 增加了复杂度

使用场景:

在这里插入图片描述

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

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

相关文章

【Python】外网远程登录访问jupyter notebook+pycharm使用ipython

第一步&#xff1a;创建python虚拟环境 conda create -n py3610 python3.6.10第二步&#xff1a;安装ipython pip install ipython pip install ipython notebook第三步&#xff1a;创建 IPython Notebook 服务器配置文件 # 进入python交互shell&#xff0c;设置密码 >&…

信钰证券|沪指震荡涨0.26%,传媒等板块拉升,消费电子概念活跃

5日早盘&#xff0c;沪指盘中窄幅震荡上扬&#xff0c;创业板指、科创50指数走高&#xff0c;北证50指数跌超2%&#xff1b;北向资金小幅流入。 截至午间收盘&#xff0c;沪指涨0.26%报3047.2点&#xff0c;深成指微涨0.05%&#xff0c;创业板指涨0.42%&#xff0c;科创50指数…

GEE 数据集 ——利用leafmap python软件包实现NASA数据的接入(colab示例)

我们如何获取我们想要的数据,这里我们通过 leafmap python软件包实现NASA数据种全球超过9000+的数据集产品的接入和使用。这里我们使用在线的colab来实现处理,因为这里我们可以很好的应用已经在线配置好的colab环境来实现,省去了安装过程的繁琐。 要下载和访问数据,您需要…

AI预测福彩3D第一弹【2024年3月4日预测】

众所周知&#xff0c;深度学习算法&#xff08;AI算法&#xff09;由于其内部含有庞大数量的神经元&#xff0c;理论上能够拟合任意维度的数据&#xff0c;目前在大数据分析领域应用非常广泛&#xff0c;并且能够很好的挖掘数据规律&#xff0c;对相关数据进行预测分析。 前面一…

Unity 角色控制(初版)

角色控制器组件&#xff0c;当然是将组件放在角色上了。 using System.Collections; using System.Collections.Generic; using UnityEngine;public class c1 : MonoBehaviour {// 获取角色控制器private CharacterController player;void Start(){// 加载角色控制器player …

【Python】进阶学习:pandas--info()用法详解

【Python】进阶学习&#xff1a;pandas–info()用法详解 &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望得到您的订…

蒙提霍尔问题

文章目录 1.简介2.答案3.直觉的错误参考文献 1.简介 蒙提霍尔问题&#xff08;Monty Hall problem&#xff09;是美国电视游戏节目《Let’s Make a Deal》中的一个问题&#xff0c;并以主持人 Monty Hall 命名。 蒙提霍尔问题也叫「三门问题」或「山羊汽车问题」。 假设您正…

超简单❗十步创建知识管理系统,效率开挂

真的想在一个行业深耕&#xff0c;建立一个知识管理系统是一件很棒的事。不光自己找起来思路清晰&#xff0c;给员工培训也是很方便的&#xff0c;很多刚入门的同事&#xff0c;在公司有详细的Sop流程情况下&#xff0c;简单培训就能上岗了。创建一个详细的知识管理系统可以按照…

Softmax 回归 + 损失函数 + 图片分类数据集【动手学深度学习v2】李沐动手学深度学习课程笔记

目录 Softmax回归 损失函数 图片分类数据集 Softmax回归从零开始实现 Softmax回归简洁实现 Softmax回归 回归和分类的区别 回归问题举例上节课的预测房价问题&#xff0c;分类问题就是对样本进行分类 回归和分类的具体区别 假设真实的类别为第i个类别&#xff08;值为1&#x…

Xcode 15 适配 MonkeyDev

升级到Xcode15后,使用Xcode创建MonkeyApp后,运行会报错,本篇文章主要讲述此过程遇到的错误和解决办法。 问题1:找不到libc++.dylib文件 问题描述: Build input files cannot be found: /usr/lib/libstdc++.dylib, /usr/lib/libc++.dylib. Did you forget to declare th…

新零售SaaS架构:订单履约系统的概念模型设计

订单履约系统的概念模型 订单&#xff1a;客户提交购物请求后&#xff0c;生成的买卖合同&#xff0c;通常包含客户信息、下单日期、所购买的商品或服务明细、价格、数量、收货地址以及支付方式等详细信息。 子订单&#xff1a;为了更高效地进行履约&#xff0c;大订单可能会被…

探索Java开发面试笔记:以听为目的,助力编程技术提升与面试准备

文章目录 一、制作背景介绍二、 Java开发面试笔记&#xff1a;为你的编程之路加速2.1 公众号主题和目标读者群体2.2 为什么面试笔记对于提高编程视野和技术至关重要2.3 親測效率 三、形式案例3.1 文章形式3.2 手机案例3.3 电脑案例 一、制作背景介绍 做公众号的背景&#xff1a…