深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南
深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南
- 前言
- 第一:AOP和代理模式
- AOP(面向切面编程):
- 代理模式:
- 第二:深入分析CGLIB代理,包括其实现原理和内部机制
- CGLIB的实现原理和内部机制:
- 使用CGLIB代理的步骤:
- 第三:探讨JDK动态代理的优点和局限性,以及何时选择它
- 优点:
- 局限性:
- 何时选择JDK动态代理:
- 第四:比较CGLIB和JDK动态代理的性能,帮助你做出明智的选择。
- JDK动态代理的性能:
- CGLIB的性能:
- 选择时的考虑:
- 第五:实际示例:如何使用CGLIB代理在Spring Boot应用程序中实现自定义事务管理。
- 第六:实际示例:如何使用JDK动态代理创建可插拔的安全性检查。
前言
在现代Java应用程序开发中,Spring Boot是一个备受欢迎的框架,它为开发人员提供了强大的工具来创建可扩展、易维护的应用程序。其中一个关键功能是AOP(面向切面编程),它允许我们在不侵入应用程序核心逻辑的情况下添加各种功能。在这篇博客中,我们将深入探讨Spring Boot中AOP的两个主要代理方式:CGLIB和JDK动态代理。我们将揭开它们的工作原理、适用场景以及如何使用它们来实现强大的切面功能。
第一:AOP和代理模式
🔗:【解锁Spring Boot AOP的魔力:优雅地管理交叉关注点】https://blog.csdn.net/Mrxiao_bo/article/details/133745778
AOP(面向切面编程):
概念:AOP是一种编程范例,旨在通过将横切关注点(cross-cutting concerns)从主要业务逻辑中分离出来,以提高代码的模块化性和可维护性。横切关注点通常包括日志记录、事务管理、安全性、异常处理等,它们跨越应用程序的多个模块和类。
原理:AOP通过将这些关注点表示为切面(Aspect)来实现。切面是一组与特定关注点相关的行为,通常通过通知(Advice)来实现,这些通知可以在程序执行的不同点被插入。这些点被称为连接点(Join Point)。切面还定义了切点(Pointcut),它决定了在何处插入通知。
代表框架:Spring框架是一个流行的Java框架,它提供了AOP的支持,允许你使用注解或XML配置来定义切面、切点和通知。
代理模式:
概念:代理模式是一种结构型设计模式,它允许你创建一个代理对象,以控制对其他对象的访问。代理对象充当客户端和目标对象之间的中介,可以用于实现懒加载、权限控制、日志记录等功能。
类型:代理模式分为两种主要类型:静态代理和动态代理。
静态代理:在编译时创建代理类,需要为每个被代理的类创建一个代理类。
动态代理:在运行时创建代理对象,通常基于接口来生成代理,无需为每个类创建代理类。
应用:代理模式广泛用于实现横切关注点,正如AOP所描述的那样。例如,代理对象可以用于记录方法调用、检查用户权限或延迟加载资源。
要为代码实现AOP和代理模式,你可以使用相关的编程库和框架,例如Spring框架和Java的动态代理机制。并且,如您所要求,要确保在代码中进行注释以提高可读性和可维护性。
第二:深入分析CGLIB代理,包括其实现原理和内部机制
CGLIB(Code Generation Library)是一个广泛用于Java中的动态代理库,通常与Spring等框架一起使用。它的主要特点是不需要目标对象实现接口,而是通过生成目标对象的子类来实现代理。下面是对CGLIB代理的深入分析,包括其实现原理和内部机制:
CGLIB的实现原理和内部机制:
-
字节码生成:CGLIB使用字节码生成技术,通过修改字节码生成代理类。这与Java的动态代理机制不同,后者使用代理接口生成代理对象。
-
目标类的子类:CGLIB创建目标类的子类,子类继承了目标类的所有非final方法。这个子类就是代理类。
-
代理方法的覆盖:CGLIB会为代理类中的每个方法生成一个覆盖方法。这个覆盖方法包含代理逻辑,例如在方法执行前后执行额外的操作,如日志记录、权限检查等。
-
方法拦截器:CGLIB使用一个方法拦截器(MethodInterceptor)来定义代理的行为。这个拦截器可以捕获方法调用,并在方法调用前后插入自定义的逻辑。
-
创建代理对象:通过CGLIB,你可以创建一个代理工厂,这个工厂用于为指定的目标对象生成代理。代理工厂会生成一个代理类的实例,并将方法调用委托给方法拦截器。
-
性能考虑:CGLIB的代理通常比JDK动态代理更快,因为它不需要通过反射来调用方法。然而,CGLIB代理在生成和加载代理类的过程中可能会稍微慢一些。
使用CGLIB代理的步骤:
-
定义一个方法拦截器,实现你的代理逻辑。
-
创建一个CGLIB代理工厂对象。
-
配置代理工厂,设置目标对象和方法拦截器。
-
通过代理工厂创建代理对象。
-
使用代理对象调用方法,代理会在方法执行前后执行方法拦截器中定义的逻辑。
总的来说,CGLIB代理通过生成字节码来创建目标类的子类,这个子类作为代理对象。代理对象覆盖了目标类的方法,以添加自定义逻辑。CGLIB的强大之处在于它可以代理没有实现接口的类,而且在某些情况下,它的性能也更好。
第三:探讨JDK动态代理的优点和局限性,以及何时选择它
JDK动态代理是Java标准库提供的一种代理机制,它具有许多优点,但也有一些局限性。以下是对JDK动态代理的优点和局限性以及何时选择它的讨论:
优点:
-
接口支持:JDK动态代理要求目标对象实现接口,这有助于遵循面向接口的编程原则,使代码更加模块化和可维护。
-
标准库支持:JDK动态代理是Java标准库的一部分,不需要额外的依赖库,因此它是Java生态系统中广泛使用的代理机制。
-
类型安全:由于代理对象遵循接口,因此在编译时具有类型安全性,编译器可以捕获许多潜在的类型错误。
-
简化:JDK动态代理相对简单,通常不需要引入复杂的字节码操作或类生成技术,易于理解和使用。
-
性能:对于简单的代理需求,JDK动态代理可以具有合理的性能,特别是在Java虚拟机的优化下。
局限性:
-
接口限制:JDK动态代理要求目标对象实现接口,这对于没有实现接口的类或已经存在的第三方库类是不适用的。
-
只能代理接口方法:JDK动态代理只能代理接口中的方法,不能代理类中的方法,这限制了其适用范围。
-
性能问题:对于复杂的代理需求或高性能要求,JDK动态代理可能不是最佳选择。因为它通过反射来调用方法,相对于直接调用目标方法,有一定的性能开销。
-
无法改变目标类:JDK动态代理无法改变目标类的继承结构,这意味着它无法添加新的成员变量或方法。
何时选择JDK动态代理:
-
当目标对象实现接口:如果你的目标对象已经实现了接口,且你只需代理接口中的方法,那么JDK动态代理是一个合适的选择。
-
当不需要复杂的代理逻辑:如果你的代理逻辑相对简单,不需要进行复杂的操作,比如横切关注点(如日志记录、事务管理)的控制,那么JDK动态代理足够。
-
当性能要求适中:对于一般的性能需求,JDK动态代理通常能够满足要求。如果性能是关键问题,可以考虑其他代理方式,如CGLIB。
总之,JDK动态代理是一个简单且方便的代理机制,适用于许多场景,特别是当目标对象已经实现接口且代理需求相对简单时。然而,如果需要更高的性能或需要代理非接口方法,可能需要考虑其他代理方式。
第四:比较CGLIB和JDK动态代理的性能,帮助你做出明智的选择。
比较CGLIB和JDK动态代理的性能可以帮助你根据具体需求做出明智的选择。这两种代理方式在性能方面各有优势和劣势。下面是对它们的性能比较:
JDK动态代理的性能:
-
性能较好:JDK动态代理在Java标准库中实现,受到了虚拟机的优化支持,因此在一般情况下具有较好的性能。
-
类型安全:由于代理对象遵循接口,编译器可以捕获许多类型错误,提供了类型安全性。
-
不需要额外依赖:JDK动态代理不需要额外的依赖库,因为它是Java标准库的一部分,这降低了项目的复杂度。
CGLIB的性能:
-
性能略低:CGLIB代理在生成子类并使用字节码技术来实现代理时,性能通常略低于JDK动态代理。这是因为字节码生成和类加载可能引入一些开销。
-
无接口要求:CGLIB代理不要求目标对象实现接口,因此更灵活。它可以代理没有接口的类。
-
支持代理类的方法:CGLIB代理可以代理类中的方法,而不仅限于接口方法,这在某些情况下非常有用。
选择时的考虑:
-
性能需求:如果性能是你的首要考虑因素,且代理的方法调用频繁且复杂,那么JDK动态代理可能是更好的选择,因为它在性能上通常优于CGLIB。
-
接口要求:如果目标对象已经实现了接口,或者你要代理的对象需要强制遵循接口,那么JDK动态代理是更适合的选择。
-
非接口方法代理:如果你需要代理非接口的类中的方法,或者目标对象没有实现接口,那么CGLIB代理是更灵活的选项。
-
项目依赖:如果你希望项目尽量减少外部依赖,JDK动态代理可能更合适,因为它是Java标准库的一部分。
总结而言,性能方面,JDK动态代理通常在一般情况下表现更好,但在一些情况下,如需要代理非接口方法或没有接口的类时,CGLIB代理提供了更大的灵活性。选择代理方式时,应根据具体需求和项目背景来决定。如果不确定,可以在项目的早期使用JDK动态代理,后期再根据性能分析和需求调整。
第五:实际示例:如何使用CGLIB代理在Spring Boot应用程序中实现自定义事务管理。
在Spring Boot应用程序中使用CGLIB代理来实现自定义事务管理是一个常见的需求,这允许你对事务进行额外的控制和管理。下面是一个实际示例,演示如何使用CGLIB代理来实现自定义事务管理:
步骤1:创建Spring Boot项目
首先,创建一个Spring Boot项目,并确保已添加Spring Boot和Spring AOP的依赖。
步骤2:创建自定义事务管理器
创建一个自定义事务管理器,这个管理器将处理事务的启动、提交和回滚。这里以简单的示例为例,展示事务管理器的基本结构:
import org.springframework.stereotype.Component;
import org.springframework.transaction.annotation.Transactional;@Component
public class CustomTransactionManager {@Transactionalpublic void beginTransaction() {// Perform transaction initialization or setup here}public void commit() {// Perform transaction commit logic}public void rollback() {// Perform transaction rollback logic}
}
步骤3:创建一个服务类
创建一个服务类,该类将使用CGLIB代理来包装事务管理器,以便在每个方法调用前后进行事务管理:
import org.springframework.aop.framework.AopProxy;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;@Service
public class TransactionalService {@Autowiredprivate CustomTransactionManager transactionManager;public void performTransactionalOperation() {// Begin a transactiontransactionManager.beginTransaction();try {// Perform some business logic here// ...// Commit the transactiontransactionManager.commit();} catch (Exception e) {// If an exception occurs, roll back the transactiontransactionManager.rollback();throw e;}}
}
步骤4:配置CGLIB代理
在Spring Boot应用程序中,你需要配置CGLIB代理以确保它在事务管理方面生效。这可以通过在配置类中添加@EnableAspectJAutoProxy
注解来完成,如下所示:
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.context.annotation.EnableAspectJAutoProxy;
import org.springframework.transaction.annotation.EnableTransactionManagement;@Configuration
@EnableAspectJAutoProxy(proxyTargetClass = true)
@EnableTransactionManagement
public class AppConfig {@Beanpublic CustomTransactionManager transactionManager() {return new CustomTransactionManager();}
}
步骤5:测试
现在,你可以编写一个测试方法来验证CGLIB代理是否成功地在事务管理中起作用:
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.boot.test.context.SpringBootTest;
import org.springframework.test.context.junit4.SpringRunner;
import org.junit.Test;
import org.junit.runner.RunWith;@RunWith(SpringRunner.class)
@SpringBootTest
public class TransactionalServiceTest {@Autowiredprivate TransactionalService transactionalService;@Testpublic void testCustomTransactionManager() {transactionalService.performTransactionalOperation();// Add assertions to verify the behavior}
}
这个示例演示了如何使用CGLIB代理来包装自定义事务管理器,以实现事务的启动、提交和回滚。这允许你在事务管理方面进行额外的控制和逻辑,适应复杂的事务需求。
第六:实际示例:如何使用JDK动态代理创建可插拔的安全性检查。
使用 JDK 的动态代理来创建可插拔的安全性检查是一种常见的设计模式,它允许您在不修改现有代码的情况下添加安全性检查逻辑。以下是一个简单的示例,演示如何使用 JDK 动态代理创建可插拔的安全性检查:
首先,我们需要创建一个接口,表示要执行安全性检查的服务:
public interface SecureService {void performOperation();
}
然后,我们创建一个实际的服务类,它实现了这个接口:
public class MySecureService implements SecureService {@Overridepublic void performOperation() {System.out.println("Executing secure operation.");}
}
接下来,我们创建一个安全性检查器,它也实现了 SecureService
接口,并在其中添加了安全性检查逻辑:
public class SecurityChecker implements SecureService {private SecureService target;public SecurityChecker(SecureService target) {this.target = target;}@Overridepublic void performOperation() {// 添加安全性检查逻辑if (isUserAuthenticated()) {target.performOperation();} else {System.out.println("Access denied. User is not authenticated.");}}private boolean isUserAuthenticated() {// 在实际应用中,您可以添加适当的身份验证逻辑return true; // 这里仅作示例,始终返回 true}
}
现在,我们将使用 JDK 的动态代理来创建一个可插拔的安全性检查器。这是如何做的:
import java.lang.reflect.Proxy;public class Main {public static void main(String[] args) {SecureService realService = new MySecureService();SecureService proxyService = (SecureService) Proxy.newProxyInstance(SecureService.class.getClassLoader(),new Class[] { SecureService.class },(proxy, method, arguments) -> {if (method.getName().equals("performOperation")) {SecurityChecker securityChecker = new SecurityChecker(realService);securityChecker.performOperation();}return null;});// 使用代理对象调用方法proxyService.performOperation();}
}
这里,我们使用 Proxy.newProxyInstance
创建了一个动态代理对象,该代理对象实现了 SecureService
接口。在代理的 invoke
方法中,我们创建了一个 SecurityChecker
对象,然后调用实际服务对象的方法,但在之前添加了安全性检查逻辑。
通过这种方式,我们实现了可插拔的安全性检查,而不需要修改原始服务的代码。这是动态代理在应用程序中的一种有用方式,特别是在添加安全性层时。注意,实际的安全性检查逻辑应该更加复杂,并根据您的应用需求进行定制。
相关文章:

深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南
深入理解Spring Boot AOP:CGLIB代理与JDK动态代理的完全指南 前言第一:AOP和代理模式AOP(面向切面编程):代理模式: 第二:深入分析CGLIB代理,包括其实现原理和内部机制CGLIB的实现原理…...

【无标题】读书笔记之《智能化社会:未来人们如何生活、相爱和思考》
《智能化社会:未来人们如何生活、相爱和思考》:Digital vs Human_ how well live, love, and think in the future ,由中信出版社于2017年06月出版。作者是澳大利亚的理查德沃特森(Richard Watson)。Richard Watson在伦敦帝国理工学院从事未来…...

华为云双十一服务器数据中心带宽全动态BGP和静态BGP区别
2023华为云双十一优惠活动中提供多款云服务器选择,需要注意的是:西南-贵阳一和华北-北京一数据中心是静态BGP带宽,其他数据中心配置全动态独享BGP带宽。 静态BGP和全动态BGP带宽有什么区别?全动态BGP网络线路可用性保障更高&…...

STM32 HAL库串口使用printf
STM32 HAL库串口使用printf 背景配置说明在usart.h中添加在usart.c中添加在工程中选中微库: 测试 背景 在我们使用CubeMX生成好STM32 HAL库工程之后,我们想使用printf函数来打印一些信息,配置如下: 配置说明 在usart.h中添加 …...

【VPX610】 青翼科技基于6U VPX总线架构的高性能实时信号处理平台
板卡概述 VPX610是一款基于6U VPX架构的高性能实时信号处理平台,该平台采用2片TI的KeyStone系列多核DSP TMS320C6678作为主处理单元,采用1片Xilinx的Virtex-7系列FPGA XC7VX690T作为协处理单元,具有2个FMC子卡接口,各个处理节点之…...

Parity 战略转型引热议,将如何推动波卡生态去中心化?
Polkadot 生态的区块链基础设施公司 Parity Technologies,最近宣布了一项重要的战略调整,即正在寻求在未来几个月内,将部分现有的市场职能转移给 Polkadot 生态系统内的多个去中心化团队,这将影响 Parity Technologies 未来几个月…...

【TES641】基于VU13P FPGA的4路FMC接口基带信号处理平台
板卡概述 TES641是一款基于Virtex UltraScale系列FPGA的高性能4路FMC接口基带信号处理平台,该平台采用1片Xilinx的Virtex UltraScale系列FPGA XCVU13P作为信号实时处理单元,该板卡具有4个FMC子卡接口(其中有2个为FMC接口)&#x…...

Spring Kafka生产者实现
需求 我们需要通过Spring Kafka库,将消息推送给Kafka的topic中。这里假设Kafka的集群和用户我们都有了。这里Kafka认证采取SASL_PLAINTEXT方式接入,SASL 采用 SCRAM-SHA-256 方式加解密。 pom.xml <dependency><groupId>org.springframew…...

手把手教你入门Three.js(初识篇)
Three.js入门篇 一、Three.js和webGL的介绍二、开发和学习环境三、 三个基本概念1. 场景Scene2. 相机Camera3. 渲染器Renderer 四、三维坐标系五、材质Material六、光源1. 点光源2. 环境光3. 平行光: 七、常见几何体八、渲染器-设置设备像素比九、渲染器-锯齿属性 一、Three.js…...

Hadoop学习总结(搭建Hadoop集群(伪分布式模式))
如果前面有搭建过Hadoop集群完全分布式模式,现在搭建Hadoop伪分布式模式可以选择直接克隆完全分布式模式中的主节点(hadoop001)。以下是在搭建过完全分布式模式下的Hadoop集群的情况进行 伪分布式模式下的Hadoop功能与完全分布式模式下的Hadoop功能相同。 一、克隆…...

人性与理性共赢,真心罐头跃过增长的山海关
在北方不少地方,黄桃罐头是一种抚慰人心的力量。从大连起家,用真材实料打动人心的真心罐头,在朝着国民品牌前进的路上,需要更透彻地洞悉“人性”。 ”人的因素影响太大。我们希望可以告别个人英雄主义,用流程来保证可…...

【Redis】Docker部署Redis数据库
Docker部署Redis数据库 1. Redis介绍2. CentOS 7 安装 & Docker 配置3. 拉取Redis 镜像、创建容器3.1 配置Docker镜像源3.2 拉取Redis 镜像3.3 容器创建 1. Redis介绍 Redis(Remote Dictionary Server ),即远程字典服务,是一个开源的使用…...

【目标跟踪】多目标跟踪测距
文章目录 前言python代码(带注释)main.pysort.pykalman.pydistance.py 结语 前言 先放效果图。目标框内左上角,显示的是目标距离相机的纵向距离。目标横向距离、速度已求出,没在图片展示。这里不仅仅实现对目标检测框的跟踪&#…...

吐血整理,服务端性能测试-Docker部署MySQL/Nginx(详细步骤)
目录:导读 前言一、Python编程入门到精通二、接口自动化项目实战三、Web自动化项目实战四、App自动化项目实战五、一线大厂简历六、测试开发DevOps体系七、常用自动化测试工具八、JMeter性能测试九、总结(尾部小惊喜) 前言 基于Docker部署My…...

基于单片机设计的智能窗帘控制系统
一、前言 智能家居技术在近年来取得了巨大的发展,并逐渐成为人们日常生活中的一部分。智能家居系统带来了便利、舒适和高效的生活体验,拥有广泛的应用领域,其中之一就是智能窗帘控制系统。 传统窗帘需要手动操作,打开或关闭窗帘…...

WSL的秘钥被修改了要怎么弄
WSL的秘钥被修改了要怎么弄 gitgithub.com: Permission denied (publickey).ssh-add -l但是我是想加到github上的guiaguaide1.github.com里面哎,为什么这个是shengyi gitgithub.com: Permission denied (publickey). git push -u origin报错 aaaASUS:~/ML/paper/A…...

cesium开发引入方式
无独有偶,引入无非两种方式:外部标签引入和import导入。 1、外部引入 外部引入的话需要提前去下载开发包,下载完后,Build文件夹有两个文件夹:Cesium和CesiumUnminified,Cesium是压缩版的,Cesiu…...

无缝的链间互操作性:通用消息传递的强大之处
前言 通用消息传递(General Message Passing,GMP)是一种支持区块链之间通信和数据传输的机制。GMP正在成为增强不同区块链网络之间互操作性的解决方案。GMP允许应用程序构建者通过使用安全消息在区块链之间通信和交换信息来利用任何区块链的…...

minio + linux + docker + spring boot实现文件上传与下载
minio docker spring boot实现文件上传与下载 1.在linux上安装并启动docker2.在docker中拉取minio并启动3.Spring Boot 整合 minio4.测试 minio 文件上传、下载及图片预览等功能 1.在linux上安装并启动docker 检查linux内核,必须是3.10以上 uname ‐r安装docker…...

vue ant DatePicker 日期选择器 限制日期可控范围
场景 限制当前日期之前不能选择 限制只能选择日期区间内 Ant Design Vue 效果 <a-date-picker :disabledDate"disabledDate"></a-date-picker>method // 限制日期选择disabledDate(current) {return current && current > moment().endOf(&…...

linux 音视频架构 linux音视频开发
linux 音视频架构 linux音视频开发 转载 mob6454cc65110a 2023-07-26 22:27:01 文章标签 linux 音视频架构 ci QT 视频教程 文章分类 架构 后端开发 阅读数117 目录 前言1、软件工具准备a. 录音软件b. 录屏软件c. 摄像头软件d. 安卓屏幕操作软件e. 视频剪辑软件2、视频教…...

el-table添加固定高度height后高度自适应
0 效果 1 添加自定义指令 新建目录src/directive/el-table 在el-table目录下新建文件adaptive.js import { addResizeListener, removeResizeListener } from element-ui/src/utils/resize-event// 设置表格高度const doResize async(el, binding, vnode) > {// 获取表格…...

Python分享之多进程探索 (multiprocessing包)
在初步了解Python多进程之后,我们可以继续探索multiprocessing包中更加高级的工具。这些工具可以让我们更加便利地实现多进程。 进程池 进程池 (Process Pool)可以创建多个进程。这些进程就像是随时待命的士兵,准备执行任务(程序)。一个进程池中可以容…...

Boris FX Mocha Pro 2023:Mac/win全能影像处理神器
Boris FX Mocha Pro 2023是一款广受欢迎的影像处理软件,它凭借其强大的功能和卓越的性能,成为了影视后期、广告制作、动画设计等领域的必备工具。无论您是专业的影视制作人员,还是初入行的新手,Boris FX Mocha Pro 2023都能为您的…...

elementUI 特定分辨率(如1920*1080)下el-row未超出一行却换行
在1920*1080分辨率下, el-col 内容未超出 el-col 宽度,el-col 不足以占据一行,el-row 却自动换行了(其他分辨率没有这个问题)。 截图: 排查: el-col 内容没有溢出;没有多余的 pad…...

mac电脑视频处理推荐:达芬奇DaVinci Resolve Studio 18 中文最新
DaVinci Resolve Studio 18是一款专业的视频编辑、调色和后期制作软件,由Blackmagic Design开发。它被广泛应用于电影、电视和广告等行业,提供了全面的工具和功能,使用户能够进行高质量的影片制作和后期处理。 以下是DaVinci Resolve Studio…...

OKLink携手CertiK在港举办Web3生态安全主题论坛
2023年10月23日,OKLink与CertiK共同发起的Web3生态安全主题论坛在香港铜锣湾拉开帷幕。本次论坛由OKLink和CertiK主办,香港投资推广署独家支持,聚焦如何构建安全可靠的Web3生态系统议题,同时深入剖析这一进程中所面临的潜在挑战。…...

王道p40 1.设计一个递归算法,删除不带头结点的单链表L中的所有值为x的结点(c语言代码实现)图解递归
视频讲解(献丑了):p40 第1题 王道数据结构课后代码题c语言代码实现_哔哩哔哩_bilibilihttps://www.bilibili.com/video/BV1Xa4y1Q7ui/?spm_id_from333.999.0.0 首先它是一个不带头结点的单链表 我们就得特殊处理 我们先让*LNULL; 然后为s开辟一个新…...

深入浅出排序算法之希尔排序
目录 1. 原理 2. 代码实现 3. 性能分析 1. 原理 希尔排序法又称缩小增量法。希尔排序法的基本思想是:先选定一个整数,把待排序文件中所有记录分成个组,所有距离为的记录分在同一组内,并对每一组内的记录进行排序。然后…...

close excel by keyword 根据关键字关闭 excel 窗口 xlwings 方式实现
根据标题关键字关闭 workbook,如果没有打开的 workbook 则退出 excel xlwings 方式实现 更方便快捷 def close_excel_by_keyword(keyword):if ~$ in keyword:returnapp xw.apps.activefor workbook in app.books:if keyword in workbook.name:workbook.close()fi…...