【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 : 代理对象,即上面的proxyObject
                            method : 对应于在代理对象上调用的接口方法的 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增强)
  • 代理模式将客户端和目标对象解耦

缺点:

  • 增加了复杂度

使用场景:

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

代码随想录第51天|● 309.最佳买卖股票时机含冷冻期 ● 714.买卖股票的最佳时机含手续费 ●总结

文章目录 ● 309.最佳买卖股票时机含冷冻期思路代码 ● 714.买卖股票的最佳时机含手续费思路&#xff1a; ●总结 ● 309.最佳买卖股票时机含冷冻期 思路 代码 class Solution {public int maxProfit(int[] prices) {// 0.买入状态-(持有)// 1.保持卖出股票的状态// 2.今天…

JVM运行时数据区——堆

文章目录 1、堆的核心概述1.1、JVM实例与堆内存的对应关系1.2、堆与栈的关系1.3、JVM堆空间划分 2、设置堆内存大小与内存溢出2.1、设置堆内存大小2.2、内存溢出案例 3、新生代与老年代4、图解对象分配过程5、Minor GC、Major GC、Full GC5.1、GC的分类5.2、分代式GC策略的触发…

Sora:AI视频模型的无限可能与挑战

随着人工智能技术的突飞猛进&#xff0c;AI视频模型已成为科技领域的新焦点。OpenAI推出的AI视频模型Sora&#xff0c;凭借其卓越的技术性能和前瞻性&#xff0c;为AI视频领域的发展揭开了新的篇章。本文将从技术解析、应用场景、未来展望、伦理与创意以及用户体验与互动五个方…

【python基础学习09课_装饰器、模块、文件】

一、项目的日志 1、日志意义与级别 1、日志的意义&#xff1a;项目的日志 -- 开发编写的&#xff0c;日志记录 -- 测试就是去查看日志信息&#xff08;为了协助我们进行问题的定位&#xff09; 可以根据日志&#xff0c;看是哪个应用的哪台机器&#xff0c;出现了什么问题&…

Scala 之舞:林浩然与杨凌芸的 IDEA 冒险

Scala 之舞&#xff1a;林浩然与杨凌芸的 IDEA 冒险 The Dance of Scala: The IDEA Adventure of Lin Haoran and Yang Lingyun 在那个阳光明媚的日子里&#xff0c;林浩然如同一位英勇的探险家&#xff0c;踏入了 Scala 的 IntelliJ IDEA 开发环境的奇妙领域&#xff0c;他带着…

day7 字符数组

1&#xff1a;输入一个字符串&#xff0c;实现单词逆置 输入:"good good study" 输出&#xff1a;"study good good" 6 //单词逆置7 // good good study8 // study good good9 10 //整体逆置11 char str[50]"good good stu…

springboot集成logback打印彩色日志

一、logback介绍 Logback是由log4j创始人设计的另一个开源日志组件,官方网站&#xff1a; logback.qos.ch。它当前分为以下三个模块&#xff1a; logback-core&#xff1a;其它两个模块的基础模块。logback-classic&#xff1a;它是log4j的一个改良版本&#xff0c;同时它完整实…

C++基于多设计模式下的同步异步日志系统day6

C基于多设计模式下的同步&异步日志系统day6 &#x1f4df;作者主页&#xff1a;慢热的陕西人 &#x1f334;专栏链接&#xff1a;C基于多设计模式下的同步&异步日志系统 &#x1f4e3;欢迎各位大佬&#x1f44d;点赞&#x1f525;关注&#x1f693;收藏&#xff0c;&am…

云手机运行在云端?安全性有保障吗

随着云计算技术的不断发展&#xff0c;云手机作为一种新兴的移动终端形态&#xff0c;逐渐成为人们关注的焦点。然而&#xff0c;对于许多人来说&#xff0c;云手机 是一个相对陌生的概念&#xff0c;安全性成为了他们最为关心的问题之一。本文将就云手机运行在云端的特点以及其…

JWT身份验证

在实际项目中一般会使用jwt鉴权方式。 JWT知识点 jwt&#xff0c;全称json web token &#xff0c;JSON Web令牌是一种开放的行业标准RFC 7519方法&#xff0c;用于在两方安全地表示声明。具体网上有许多文章介绍&#xff0c;这里做简单的使用。 1.数据结构 JSON Web Token…

#include<ros/ros.h>头文件报错

快捷键 ctrl shift B 调用编译&#xff0c;选择:catkin_make:build&#xff09;(要先在vscode上添加扩展&#xff1a;ros) 可以点击配置设置为默认&#xff0c;修改.vscode/tasks.json 文件 修改.vscode/tasks.json 文件&#xff0c;否则ros.h头文件会报错 内容修改为以下内…

鸿蒙Harmony应用开发—ArkTS声明式开发(通用属性:拖拽控制)

设置组件是否可以响应拖拽事件。 说明&#xff1a; 从API Version 10开始支持。后续版本如有新增内容&#xff0c;则采用上角标单独标记该内容的起始版本。 ArkUI框架对以下组件实现了默认的拖拽能力&#xff0c;支持对数据的拖出或拖入响应&#xff0c;开发者只需要将这些组件…

残差网宽度残差网

目录 from # 从ResNet说起 # 更深的网络 # 理解shortcut # WRN之宽度残差块 # 退化现象 from WideResNet(宽残差网络)算法解析-鸟类识别分类-Pytorch实战-CSDN博客 # 前言 ResNet可以训练出更深的CNN模型ResNet模型的核心是建立前面层和后面层之间“短路链接”&#xff…

【Proteus仿真】【Arduino单片机】坐姿矫正提醒器设计

文章目录 一、功能简介二、软件设计三、实验现象联系作者 一、功能简介 本项目使用Proteus8仿真Arduino单片机控制器&#xff0c;使用LCD1602液晶显示模块、HC-SR04超声波模块、蜂鸣器、按键、人体红外传感器等。 主要功能&#xff1a; 系统运行后&#xff0c;LCD1602显示超声…

java——2024-03-03

String类的对象能被修改吗&#xff1f;如果不能需要用什么修改&#xff1f;StringBuilder和StringBuffer的区别&#xff1f;equals和区别谈谈对面向对象的理解重载和重写的区别说一下ArrayList&#xff0c;LinkedList底层实现以及区别什么是哈希冲突&#xff1f;hashMap和conCu…

tomcat动静分离和负载均衡

目录 引言 1.实验环境搭建 2.部署Nginx服务器及配置静态页面Web服务 3.部署Tomcat服务及配置动态页面Web服务 4.实验验收 动态页面 静态页面 引言 tomcat服务既可以处理动态页面&#xff0c;也可以处理静态页面&#xff1b;但其处理静态页面的速度远远不如nginx和apache…

笨办法学 Python3 第五版(预览)(三)

原文&#xff1a;Learn Python the Hard Way, 5th Edition (Early Release) 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 练习 30&#xff1a;假如 这是你将要输入的下一个 Python 脚本&#xff0c;它向你介绍了if语句。输入这个代码&#xff0c;确保它能够完美运行…

gin gorm学习笔记

代码仓库 https://gitee.com/zhupeng911/go-advanced.git https://gitee.com/zhupeng911/go-project.git 1. gin介绍 Gin 是使用纯 Golang 语言实现的 HTTP Web框架&#xff0c;Gin接口设计简洁&#xff0c;提供类似Martini的API&#xff0c;性能极高&#xff0c;现在被广泛使用…

基于ZYNQ PS-SPI的Flash驱动开发

本文使用PS-SPI实现Flash读写&#xff0c;PS-SPI的基础资料参考Xilinx UG1085的文档说明&#xff0c;其基础使用方法是&#xff0c;配置SPI模式&#xff0c;控制TXFIFO/RXFIFO&#xff0c;ZYNQ的IP自动完成发送TXFIFO数据&#xff0c;接收数据到RXFIFO&#xff0c;FIFO深度为12…

Vmware Workstation 不可恢复错误:0xc0000005 has occured

上周打开虚拟机的时候报错&#xff1a;Vmware Workstation 不可恢复错误&#xff1a;0xc0000005 has occured&#xff0c;查看网上资料说是vmware版本太低&#xff0c;需要手动更新本地版本。 由于本地网络不是很好&#xff0c;没能正常更新&#xff0c;无意中出现问题前更改了…