Spring-4-代理

前面提到过,在Spring中有两种类型的代理:使用JDK Proxy类创建的JDK代理以及使用CGLIB Enhancer类创建的基于CGLIB的代理。

你可能想知道这两种代理之间有什么区别,以及为什么 Spring需要两种代理类型。

在本节中,将详细研究代理之间的差异。
代理的核心目标是拦截方法调用,并在必要时执行适用于特定方法的通知链。

通知的管理和调用基本上是独立于代理的,由SpringAOP框架管理。

而代理主要负责拦截对所有方法的调用,并将它们根据需要传递给AOP框架,以便应用通知。

除上述核心功能外,代理还必须支持一组附加功能。可以通过AopContext类(这是一个抽象类)配置代理以公开自己,以便可以检索代理并从目标对象调用代理上的被通知方法。

当通过ProxyFactorysetExposeProxy()启用该功能时,代理负责确保代理类被适当地公开。

另外,所有代理类默认实现Advised接口,从而允许在创建代理之后更改通知链。

代理还必须确保任何返回代理类(即返回代理目标)的方法实际上返回的是代理而不是目标。

正如你所看到的,典型的代理需要执行很多工作,并且所有这些逻辑都在JDK和CGLIB代理中实现。

使用JDK动态代理

JDK代理是Spring中最基本的代理类型。

与CGLIB代理不同,JDK代理只能生成接口的代理,而不能生成类的代理。

这样一来,想要代理的任何对象都必须至少实现一个接口,并且生成的代理将是实现该接口的对象。

在这里插入图片描述

一般来说,为类使用接口是一种很好的设计,但并不总是可行的,尤其是当使用第三方或旧代码时。

在这种情况下,必须使用CGLIB代理。当使用JDK代理时,所有方法调用都会被JVM拦截并路由到代理的invoke()方法。

然后由invoke()方法确定是否通知有关方法(根据由切入点定义的规则),如果确定想要通知,则通过使用反射调用通知链,然后调用方法本身。

在调用invoke()之前,JDK代理无法区分被通知方法和未被通知方法。

这意味着对于代理上的未被通知方法,invoke()方法仍然会被调用,所有检查仍然会执行,并且仍然可以通过使用反射进行调用。

显然,每次调用方法时,都会导致运行时开销,即使代理不会执行额外的处理,而只是通过反射调用未被通知的方法。

使用CGLIB代理

如果使用JDK代理,那么在每次调用invoke()方法时,有关如何处理特定方法调用的决策都会在运行时做出。

而当使用CGLIB时,CGLIB会为每个代理动态生成新类的字节码,并尽可能重用已生成的类。

在这种情况下,所生成的代理类型将是目标对象类的子类。

在这里插入图片描述
当首次创建CGLIB代理时,CGLIB会询问Spring如何处理每个方法。这意味着每次调用JDK代理上的invoke()时所执行的许多决策对于CGLIB代理来说只会执行一次。

由于CGLIB生成实际的字节码,因此在处理方法的方式上有更多的灵活性。

例如,CGLIB代理可以生成适当的字节码来直接调用任何未被通知的方法,从而减少代理所带来的开销。

另外,CGLIB代理可以确定一个方法是否返回代理,如果不返回,则允许直接调用方法调用,从而进一步减少运行时间开销。

CGLIB代理还以不同于JDK代理的方式处理固定通知链。

固定通知链是在代理生成后不会更改的链。默认情况下,即使在创建代理后,也可以更改代理上的顾问和通知,虽然很少有必要这么做。

CGLIB代理以特定方式处理固定通知链,从而减少执行通知链的运行时间开销。

在这里插入图片描述
具有冻结通知链(frozen advice chain)的CGLIB代理(即,当通过调用ProxyConfig类中的setFrozen()方法来冻结代理时,CGLIB将执行进一步的优化,但是,不允许更改通知)。

选择要使用的代理

决定使用哪个代理通常很容易。CGLIB代理可以代理类和接口,而JDK代理只能代理接口。

在性能方面,除非在冻结模式下使用CGLIB,否则JDK和CGLIB标准模式之间没有显著差异(至少在运行被通知和未被通知方法时没有显著差异)。

在这种情况下,通知链不能更改且CGLIB在冻结模式下会进行进一步优化。

当需要代理类时,CGLIB代理是默认选择,因为它是唯一能够生成类代理的代理。

如果想要在代理接口时使用CGLIB代理,必须使用setOptimize()方法将ProxyFactory中的optimize标志的值设置为true。

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

如若内容造成侵权/违法违规/事实不符,请联系一条长河网进行投诉反馈,一经查实,立即删除!

相关文章

基于策略模式和简单工厂模式实现zip、tar、rar、7z四种压缩文件格式的解压

推荐语 这篇技术文章深入探讨了基于策略模式和简单工厂模式实现四种常见压缩文件格式的解压方法。通过阅读该文章,你将了解到如何利用这两种设计模式来实现灵活、可扩展的解压功能,同时适应不同的压缩文件格式。如果你对设计模式和文件处理感兴趣或刚好…

【JavaEE进阶】 @RequestMapping注解

文章目录 🎄什么是RequestMapping 注解🌳RequestMapping 使⽤🌲RequestMapping 是GET还是POST请求?🚩使用Postman构造POST请求 ⭕总结 🎄什么是RequestMapping 注解 在Spring MVC 中使⽤ RequestMapping 来…

基于Mapify的在线艺术地图设计

地图是传递空间信息的有效载体,更加美观、生动的地图产品也是我们追求目标。 那么,我们如何才能制出如下图所示这样一幅艺术性较高的地图呢?今天我们来一探究竟吧! 按照惯例,现将网址给出: https://www.m…

PathWave Device Modeling (IC-CAP) 建模系统——IC-CAP概述

建模系统 PathWave Device Modeling(IC-CAP)建模系统用于测量半导体器件并分析器件的电路建模特性以及分析所得数据。 要使用PathWave Device Modeling(IC-CAP),您需要以下设置: 一台工作站执行直流、电…

【rosbag】rosbag命令--查看提取bag中的内容

1.查看消息 rosbag info --.bag topics是其中的话题内容,/imu中只写了包含imu的数据,但是imu是9轴的,到底磁力计信息是否包含在其中呢? 2.将bag中的某个话题转为txt 将file_name.bag文件中topic_name话题的消息转换到Txt_name.…

java中的缓冲类HeapByteBuffer和DirectByteBuffer的区别

使用之前写的文章里的例子 https://blog.csdn.net/zlpzlpzyd/article/details/135292683 HeapByteBuffer import java.io.File; import java.io.FileInputStream; import java.io.Serializable; import java.nio.ByteBuffer; import java.nio.channels.FileChannel;public clas…

Linux升级指南:保持系统安全和高效运行

Linux系统的升级是确保系统稳定和安全性的重要步骤。本文将介绍Linux系统升级的基本概念,以及具体的操作步骤和注意事项,以帮助用户顺利升级他们的Linux系统。 Linux操作系统以其稳定性和可定制性而闻名,它经常通过升级来提供新的功能、修复漏…

【论文笔记】Radar Fields: An Extension of Radiance Fields to SAR

原文链接:https://arxiv.org/abs/2312.12961 1. 引言 本文针对合成孔径雷达(SAR)的3D重建,提出雷达场,基于多个SAR对场景的测量学习体积模型。 3. 辐射场的介绍 NeRF将静态场景表达为连续的体积函数 F \mathcal{F}…

服装店收银系统 助力完善线上线下方案

一个服装店收银系统可以助力完善线上线下方案,提供以下功能和优势: 1. 销售管理:收银系统可以记录每笔销售订单的详细信息,包括商品名称、价格、数量等,方便店主进行销售统计和分析。 2. 库存管理:收银系统…

Android 理解Context

文章目录 Android 理解ContextContext是什么Activity能直接new吗? Context结构和源码一个程序有几个ContextContext的作用Context作用域获取ContextgetApplication()和getApplicationContext()区别Context引起的内存泄露错误的单例模式View持有Activity应用正确使用…

【C语言】数据结构——带头双链表实例探究

💗个人主页💗 ⭐个人专栏——数据结构学习⭐ 💫点击关注🤩一起学习C语言💯💫 目录 导读:1. 双链表结构特征2. 实现双向循环链表2.1 定义结构体2.2 创造节点2.3 双向链表初始化2.4 双向链表打印2…

【数据库系统概论】第7章-数据库设计

文章目录 7.1 数据库设计概述7.2 需求分析7.2.1 需求分析的任务7.2.2 需求分析的难点7.2.2 需求分析的方法7.2.3 数据字典 7.3 概念结构设计7.3.1 概念模型7.3.2 E-R模型7.3.3 概念结构设计 7.4 逻辑结构设计7.4.1 E-R图向关系模型的转换7.4.2 数据模型的优化7.4.3 设计用户子模…

PowerShell Instal 一键部署gitea

gitea 前言 Gitea 是一个轻量级的 DevOps 平台软件。从开发计划到产品成型的整个软件生命周期,他都能够高效而轻松的帮助团队和开发者。包括 Git 托管、代码审查、团队协作、软件包注册和 CI/CD。它与 GitHub、Bitbucket 和 GitLab 等比较类似。 Gitea 最初是从 Gogs 分支而来…

QT上位机开发(倒计时软件)

【 声明:版权所有,欢迎转载,请勿用于商业用途。 联系信箱:feixiaoxing 163.com】 倒计时软件是生活中经常遇到的一种场景。比如运动跑步,比如学校考试,比如论文答辩等等,只要有时间限制规定的地…

21 UVM printer

uvm_printer 类提供了以不同格式打印 uvm_objects 的灵活性。我们已经讨论了使用 uvm_field_* 宏的 print() 方法,或者如果不使用 utils_begin/ end 宏,则编写 do_print() 方法。 UVM printer提供四种内置printer。 uvm_printeruvm_table_printeruvm_t…

Git:远程仓库的使用

查看当前的远程库 要查看当前配置有哪些远程仓库,可以用git remote 命令,它会列出每个远程库的简短名字。在克隆完某个项目后,至少可以看到一个名为origin 的远程库,Git 默认使用这个名字来标识你所克隆的原始仓库: 也…

DM达梦数据库表占用空间大小

问题描述: 项目涉及用户量大且数据量大,为提高查询性能采用分表方式处理数据;根据业务要求总共4张业务表,每张业务表扩展成100张表,系统中总共400张表。部署至测试环境发现测试环境占用的磁盘空间是开发环境的8倍。 问…

从程序员到项目经理

前言 看到这个话题,博主不禁有感而发。那么就简单讲讲,从程序员到项目经理,需要具备哪些必备能力或基本的素养。 一、必备 1、技术能力 程序员首先必须成为一个名合格的coder,有思想有见解有态度。 无论身处什么开发岗位&…

python2.x编码Unicode字符串

1 python2.x编码Unicode字符串 python2.x默认编码方法为ASCII码。字符串赋值时按系统默认编码自动编码,通过decode()方法解码为Unicode,再通过encode()方法编码为指定码。 1.1 编码解码基础知识 1.1.1 位 位(bit)是计算机存储数据的最小单位&#xf…

数据之光:乡镇企业的发展利器——数据可视化

数据可视化是一项强大的工具,它不仅在大型企业中发挥关键作用,而且在乡镇企业中也能作出显著贡献。那么,数据可视化究竟能为乡镇企业做出什么样的贡献呢? 首先,数据可视化为乡镇企业提供了更清晰的业务洞察。通过将庞大…