当前位置: 首页 > news >正文

[Spring AOP 7] 动态通知调用

动态通知调用

本笔记基于黑马程序员 Spring高级源码解读

更美观清晰的版本在:Github

注意:阅读本章内容之前,建议先熟悉静态通知调用的内容

在静态通知调用一节中,我们还有一个悬而未决的问题。
我们谈到了调用链MethodInvocation中的proceed方法:

// ReflectiveMethodInvocation.java
@Override
@Nullable
public Object proceed() throws Throwable {
// 1. 如果拦截器(通知)已经递归地调用完毕,则直接调用目标方法(我们讲的是target,这里用joinpoint表示)
// 结束递归
if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {
return invokeJoinpoint();
}// 2. 否则,取下一个拦截器或动态匹配器(我们先不管什么是动态匹配器)Object interceptorOrInterceptionAdvice =this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);// 3. 如果是动态匹配器,运行下面的逻辑(动态匹配器的部分我们跳过)if (/*...*/) {/*...*/}// 4. 如果是拦截器(通知),则调用invoke方法else {return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);}
}

我们当时省略了第2、3步逻辑的讲解,现在是时候具体看看什么是“动态匹配器”了。

1. 准备工作

我们先准备好以下代码:

  • 一个含有两个前置通知的@Aspect高级切面
  • 一个Target目标类
  • 将两个bean注入

main方法的逻辑是:

  • 获取并初始化一个bean容器
  • 获取之前注入的两个bean
  • 通过AnnotationAwareAspectJAutoProxyCreator#findEligibleAdvisors获得一个切面列表
  • 创建代理工厂以及获取代理对象
  • 通过代理工厂获取通知类型
public class DynamicInvocationOfAdvice {@Aspectstatic class MyAspect {@Before("execution(* foo(..))") // 静态通知调用,不带参数绑定,执行时不需要切点public void before1() {System.out.println("before 1");}// 动态通知调用,需要参数绑定,性能更低,执行时仍需要切点// 静态部分在代理创建时就已经筛掉不可能的目标方法// 动态部分时会需要通过MethodMather.matches方法再跑一次,只有返回true才真正执行before2@Before("execution(* foo(..)) && args(x)")public void before2(int x) {System.out.printf("before 2 %d\n", x);}}static class Target {public void foo(int x) {System.out.printf("target foo(%d)%n", x);}}@Configurationstatic class MyConfig {@BeanAnnotationAwareAspectJAutoProxyCreator proxyCreator() {return new AnnotationAwareAspectJAutoProxyCreator();}@Beanpublic MyAspect myAspect() {return new MyAspect();}}@SuppressWarnings("unchecked")public static void main(String[] args) throws Throwable {GenericApplicationContext context = new GenericApplicationContext();context.registerBean(ConfigurationClassPostProcessor.class);context.registerBean(MyConfig.class);context.refresh();AnnotationAwareAspectJAutoProxyCreator creator = context.getBean(AnnotationAwareAspectJAutoProxyCreator.class);Method findEligibleAdvisors = creator.getClass().getSuperclass().getSuperclass().getDeclaredMethod("findEligibleAdvisors", Class.class, String.class);findEligibleAdvisors.setAccessible(true);List<Advisor> list = ((List<Advisor>) findEligibleAdvisors.invoke(creator, Target.class, "target"));Target target = new Target();ProxyFactory proxyFactory = new ProxyFactory();proxyFactory.setTarget(target);proxyFactory.addAdvisors(list);Target proxy = ((Target) proxyFactory.getProxy());List<Object> interceptorsList = proxyFactory.getInterceptorsAndDynamicInterceptionAdvice(Target.class.getMethod("foo", int.class), Target.class);interceptorsList.forEach(System.out::println);}
}

注意,@Aspect高级切面中的第二个前置方法是动态增强的:@Before("execution(* foo(..)) && args(x)"),这意味着切点不仅会根据方法的名字进行匹配,还会在运行时检查参数列表,根据参数的数量去匹配那些只有一个参数的方法。

我们可以打印一下结果,先看看通知(回顾:通知的本质是拦截器)都有哪些种类:

org.springframework.aop.interceptor.ExposeInvocationInterceptor@5340477f
org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor@47caedad
InterceptorAndDynamicMethodMatcher[interceptor=org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor@28cda624, matcher=AspectJExpressionPointcut: (int x) execution(* foo(..)) && args(x)]

前两个我们并不陌生:一个是外围的最大通知ExposeInvocationInterceptor,还有一个是第一个前置通知,只不过被转换为了环绕通知MethodBeforeAdviceInterceptor。这些都是我们已经见到过的静态通知。
但是第三个InterceptorAndDynamicMethodMatcher,我们点进源码中会发现它并没有实现MethodInterceptor接口,说明并不是被转换成了环绕通知。这个新的类型其实就对应着我们现在要深入的动态通知。

2. InterceptorAndDynamicMethodMatcher

虽然InterceptorAndDynamicMethodMatcher没有实现MethodInterceptor,但是MethodInterceptor是它的一个成员变量。同时它还有另一个成员变量MethodMatcher

record InterceptorAndDynamicMethodMatcher(MethodInterceptor interceptor, MethodMatcher matcher) {}

record我们就当成是带有getter和setter还有其他一些默认方法的class,这里不会深入讲解这个关键字和特性。

MethodMatcher乍一看好像没见过,实际上我们之前使用过的切点类AspectJExpressionPointcut,它所继承的接口IntroductionAwareMethodMatcherMethodMatcher的一个子接口。
所以这个MethodMatcher就是切点。也就是说,这个“拦截器与动态方法匹配器”是由一个环绕通知和一个切点组成的。这个组成很好理解,因为@Before("execution(* foo(..)) && args(x)")被拆成两部分后,前一部分需要被转换成一个环绕通知,而后一部分需要在运行时去切入方法。

我们针对InterceptorAndDynamicMethodMatcher打印一下里面的细节:

public static void main(String[] args) throws Throwable {// 以上均不变,将下面的forEach中的方法从打印变成showDetailsinterceptorsList.forEach(DynamicInvocationOfAdvice::showDetails);
}public static void showDetails(Object o) {try {Class<?> clazz = Class.forName("org.springframework.aop.framework.InterceptorAndDynamicMethodMatcher");if (clazz.isInstance(o)) {Field matcher = clazz.getDeclaredField("matcher");Field interceptor = clazz.getDeclaredField("interceptor");matcher.setAccessible(true);interceptor.setAccessible(true);System.out.println("环绕通知和切点:" + o);System.out.println("\t切点为:" + matcher.get(o));System.out.println("\t通知为" + interceptor.get(o));} else {System.out.println("普通环绕通知:" + o);}} catch (Exception e) {e.printStackTrace();}
}
普通环绕通知:org.springframework.aop.interceptor.ExposeInvocationInterceptor@7857fe2
普通环绕通知:org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor@6f15d60e
环绕通知和切点:InterceptorAndDynamicMethodMatcher[interceptor=org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor@446a1e84, matcher=AspectJExpressionPointcut: (int x) execution(* foo(..)) && args(x)]切点为:AspectJExpressionPointcut: (int x) execution(* foo(..)) && args(x)通知为org.springframework.aop.framework.adapter.MethodBeforeAdviceInterceptor@446a1e84

可以看到,确实是由一个表达式切点和一个前置通知转成的环绕通知组成。

3. 完整的proceed

现在我们可以来看看完整的proceed方法到底是怎么回事了。

@Override
@Nullable
public Object proceed() throws Throwable {// We start with an index of -1 and increment early.if (this.currentInterceptorIndex == this.interceptorsAndDynamicMethodMatchers.size() - 1) {return invokeJoinpoint();}Object interceptorOrInterceptionAdvice =this.interceptorsAndDynamicMethodMatchers.get(++this.currentInterceptorIndex);if (interceptorOrInterceptionAdvice instanceof InterceptorAndDynamicMethodMatcher dm) {// Evaluate dynamic method matcher here: static part will already have// been evaluated and found to match.Class<?> targetClass = (this.targetClass != null ? this.targetClass : this.method.getDeclaringClass());if (dm.matcher().matches(this.method, targetClass, this.arguments)) {return dm.interceptor().invoke(this);}else {// Dynamic matching failed.// Skip this interceptor and invoke the next in the chain.return proceed();}}else {// It's an interceptor, so we just invoke it: The pointcut will have// been evaluated statically before this object was constructed.return ((MethodInterceptor) interceptorOrInterceptionAdvice).invoke(this);}
}

当我们执行proxy.foo()的时候,ReflectiveMethodInvocation#proceed就会被调用。逻辑如下:

  1. 先判断是不是递归终点(即目标方法),如果是则退出递归
  2. 判断通知是动态通知还是静态通知
    1. 如果是动态通知且匹配上了方法,调用动态拦截器的invoke
    2. 如果是动态通知但是没匹配上,则跳过
  3. 如果通知是静态通知,那么直接调用静态通知拦截器的invoke

可以看出,其实逻辑跟静态通知调用是非常相似的,只不过是多了一个匹配器而已。

相关文章:

[Spring AOP 7] 动态通知调用

动态通知调用 本笔记基于黑马程序员 Spring高级源码解读 更美观清晰的版本在&#xff1a;Github 注意&#xff1a;阅读本章内容之前&#xff0c;建议先熟悉静态通知调用的内容 在静态通知调用一节中&#xff0c;我们还有一个悬而未决的问题。 我们谈到了调用链MethodInvocation…...

Redis 集群

集群基本介绍 广义的集群&#xff0c;只要是多个机器构成了分布式系统&#xff0c;都可以称为是一个“集群”&#xff08;主从模式&#xff0c;哨兵模式&#xff09; 狭义的集群&#xff0c;Redis 提供的集群模式&#xff0c;在这个模式下主要解决的是存储空间不足的问题&…...

【Redis】基础命令数据结构

文章目录 基础命令keysexistsdelexpirettltype 数据结构和内部编码 在介绍数据类型前先介绍一下 Redis 的基础命令&#xff0c;方便理解 基础命令 keys 返回所有满足样式&#xff08;pattern&#xff09;的 key keys pattern 当前有如下 key PS&#xff1a;实际开发环境和生…...

C++(6):逻辑运算符

目录 1. 代码示例 示例 1&#xff1a;基础用法 示例 2&#xff1a;条件判断 2. 短路求值&#xff08;Short-Circuit Evaluation&#xff09; 代码示例 3. 实际应用场景 场景 1&#xff1a;输入合法性验证 场景 2&#xff1a;游戏状态判断 4. 注意事项 逻辑运算符用于组…...

项目管理从专家到小白

敏捷开发 Scrum 符合敏捷开发原则的一种典型且在全球使用最为广泛的框架。 三个角色 产品负责人Product Ower&#xff1a;专注于了解业务、客户和市场要求&#xff0c;然后相应地确定工程团队需要完成的工作的优先顺序。 敏捷教练Scrum Master&#xff1a;确保 Scrum 流程顺…...

AI绘画灵感觉醒指南:从灵感源泉到创意输出

目录 一、引言 二、灵感来源 2.1 现实生活 2.2 其他艺术作品 2.3 文学作品 三、灵感转化为输入提示 3.1 明确主题与核心元素 3.2 细化描述 3.3 使用修饰词与专业术语 3.4 组合与优化提示词 四、案例分析 4.1 从现实生活获取灵感 4.2 从其他艺术作品获取灵感 4.3 …...

【Java学习】枚举(匿名类详解)

目录 一、匿名类 1.形式 2.性质 2.1匿名性 2.1.1同步性 使用场景 2.1.2复用性 2.1.3向上转型 2.2实现性 3.传参 3.1构造传全参 3.1.1过程 3.1.2效果 2.1.4原子类构造无参 4.权限 二、枚举类 1.枚举常量 2.性质 2.1多态性 2.2单例性 2.2.1private保护 2.2…...

力扣题解:2、两数相加

个人认为&#xff0c;该题目可以看作合并两个链表的变种题&#xff0c;本题与21题不同的是&#xff0c;再处理两个结点时&#xff0c;对比的不是两者的大小&#xff0c;而是两者和是否大于10&#xff0c;加法计算中大于10要进位&#xff0c;所以我们需要声明一个用来标记是否进…...

PyTorch API 9 - masked, nested, 稀疏, 存储

文章目录 torch.randomtorch.masked简介动机什么是 MaskedTensor&#xff1f; 支持的运算符一元运算符二元运算符归约操作查看与选择函数 torch.nested简介构造方法数据布局与形状支持的操作查看嵌套张量的组成元素填充张量的相互转换形状操作注意力机制 与 torch.compile 的配…...

linux测试硬盘读写速度

#!/bin/bash # 文件名&#xff1a; disk_rate.sh # linux测试硬盘读写速度 TEST_FILE"disk_speed_test.tmp" TEST_SIZE"1024M" echo "开始测试磁盘写入速度..." WRITE_RESULT$(dd if/dev/zero of$TEST_FILE bs$TEST_SIZE count1 oflagdirect 2…...

单片机系统设计不同开发方式的优缺点(面包板,洞洞板,PCB板)

目录 快速验证代码逻辑 涉及具体电路较多 涉及高频电路 快速验证代码逻辑 面包板&#xff0c;无焊接&#xff0c;适合快速搭建临时电路。优点应该是使用方便&#xff0c;不需要焊接&#xff0c;可以随时更换元件。但缺点可能是不稳定&#xff0c;接触不良&#xff0c;不适合高…...

数字信号处理|| 快速傅里叶变换(FFT)

一、实验目的 &#xff08;1&#xff09;加深对快速傅里叶变换&#xff08;FFT&#xff09;基本理论的理解。 &#xff08;2&#xff09;了解使用快速傅里叶变换&#xff08;FFT&#xff09;计算有限长序列和无限长序列信号频谱的方法。 &#xff08;3&#xff09;掌握用MATLA…...

基于CNN卷积神经网络的带频偏QPSK调制信号检测识别算法matlab仿真

目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 2.算法运行软件版本 matlab2024b 3.部分核心程序 &#xff08;完整版代码包含详细中文注释和操作步骤视频&#xff09…...

数据库索引详解:原理 · 类型 · 使用 · 优化

在关系型数据库中&#xff0c;索引&#xff08;Index&#xff09;是提高查询性能的利器。合理设计和使用索引&#xff0c;可以极大地减少 IO 操作&#xff0c;提升查询效率&#xff1b;但滥用或误用索引&#xff0c;却可能带来维护开销和性能瓶颈。我将从以下几个方面&#xff…...

Java大数据可视化在城市空气质量监测与污染溯源中的应用:GIS与实时数据流的技术融合

随着城市化进程加速&#xff0c;空气质量监测与污染溯源成为智慧城市建设的核心议题。传统监测手段受限于数据离散性、分析滞后性及可视化能力不足&#xff0c;难以支撑实时决策。2025年4月27日发布的《Java大数据可视化在城市空气质量监测与污染溯源中的应用》一文&#xff0c…...

【基于 LangChain 的异步天气查询3】OpenWeather实现实时天气查询

目录 一、项目功能概述 1、城市识别&#xff08;GeoNames API&#xff09; 2、天气数据获取&#xff08;OpenWeather API&#xff09; 3、AI 分析天气&#xff08;deepseek-r1&#xff09; 4、异步运行支持 5、配置文件隔离&#xff08;.env&#xff09; 二、注册 OpenW…...

.Net HttpClient 管理客户端(初始化与生命周期管理)

HttpClient 初始化与生命周期管理 HttpClient 旨在实例化一次&#xff0c;并在应用程序的整个生命周期内重复使用。 为实现复用&#xff0c;HttpClient类库默认使用连接池和请求管道&#xff0c;可以手动管理(连接池、配置管道、使用Polly); 结合IoC容器、工厂模式(提供了IHt…...

树莓派4的v4l2摄像头(csi)no cameras available,完美解决

根据2025年最新技术文档和树莓派官方支持建议&#xff0c;no cameras available错误通常由驱动配置冲突或硬件连接问题导致。以下是系统化解决方案&#xff1a; 一、核心修复步骤 强制禁用传统驱动 sudo nano /boot/firmware/config.txt确保包含以下配置&#xff08;2025年新版…...

VBA将PDF文档内容逐行写入Excel

VBA是无法直接读取PDF文档的&#xff0c;但结合上期我给大家介绍了PDF转换工具xpdf-tools-4.05&#xff0c;先利用它将PDF文档转换为TXT文档&#xff0c;然后再将TXT的内容写入Excel&#xff0c;这样就间接实现了将PDF文档的内容导入Excel的操作。下面的代码将向大家演示如何实…...

【STM32 学习笔记】USART串口

注意&#xff1a;在串口助手的接收模式中有文本模式和HEX模式两种模式&#xff0c;那么它们有什么区别&#xff1f;   文本模式和Hex模式是两种不同的文件编辑或浏览模式&#xff0c;不是完全相同的概念。文本模式通常是指以ASCII编码格式表示文本文件的编辑或浏览模式。在文…...

位图布隆过滤器

1.位图 所谓位图&#xff0c;就是用每一位来存放某种状态&#xff0c;适用于海量数据&#xff0c;整数&#xff0c;数据无重复的场景。通常是用来判 断某个数据存不存在的。 如上例子&#xff0c;10个整数本应该存放需要40个字节&#xff0c;此时用位图只需要3个字节。 下面代…...

【Web】使用Vue3开发鸿蒙的HelloWorld!

文章目录 1、简介2、效果3、环境3.1、开发环境3.2、运行环境 4、实现4.1、在VSCode上使用Vue开发HelloWorld4.1.1、通过 Vite 快速创建项目4.1.2、修改 src/App.vue4.1.3、模拟Web浏览器运行 4.2、使用DevEco完成手机App端移植4.2.1、构建Vue 3项目为静态文件4.2.2、创建Harmon…...

cv_area_center()

主题 用opencv实现了halcon中area_center算子的功能&#xff0c; 返回region的面积&#xff0c;中心的行坐标和中心的列坐标 代码很简单 def cv_area_center(region):area[]row []col []for re in region:retval cv2.moments(re)area.append(retval[m00])row.append(int(r…...

Python+OpenCV实现手势识别与动作捕捉:技术解析与应用探索

引言&#xff1a;人机交互的新维度 在人工智能与计算机视觉技术飞速发展的今天&#xff0c;手势识别与动作捕捉技术正逐步从实验室走向大众生活。通过Python的OpenCV库及MediaPipe等工具&#xff0c;开发者能够以较低门槛实现精准的手部动作识别&#xff0c;为虚拟现实、智能家…...

【llama-factory】Lora微调和DPO训练

微调参考 DPO参考 llama-factory官网 LoRA微调 数据集处理 数据集格式和注册 Alpaca数据集格式&#xff1a; [{"instruction": "人类指令&#xff08;必填&#xff09;","input": "人类输入&#xff08;选填&#xff09;","…...

JS较底层的用法,几类简单介绍

JS较底层的用法 在 JavaScript 中&#xff0c;“偏底层”的用法通常是指更接近语言核心、规范、底层机制的特性。这些用法不是日常开发中最常见的&#xff0c;但对理解语言原理、优化性能或构建框架、库非常重要。下面是一些常见的“偏底层”用法或特性 1. 对象属性底层操作&am…...

当可视化遇上 CesiumJS:突破传统,打造前沿生产配套方案

CesiumJS 技术基础介绍 CesiumJS 是一款基于 JavaScript 的开源库&#xff0c;专门用于创建动态、交互式的地理空间可视化。它利用 WebGL 技术&#xff0c;能够在网页浏览器中流畅地渲染高分辨率的三维地球和地图场景。CesiumJS 支持多种地理空间数据格式&#xff0c;包括但不…...

使用python脚本连接SQL Server数据库导出表结构

一. 准备工作 Mac 系统安装freetds brew install freetds 安装pymssql pip3 install pymssql 二.导出指定表的结构&#xff1a; import pymssql# 配置数据库连接参数&#xff08;根据实际情况修改&#xff09; server # 内网服务器地址或IP database # 数据库名称…...

Docker基础入门

Docker核心概念 Docker 可以让开发者打包他们的应用以及依赖包到一个轻量级、可移植的容器中&#xff0c;然后发布到任何流行的 Linux 机器上&#xff0c;也可以实现虚拟化。 容器是完全使用沙箱机制&#xff0c;相互之间不会有任何接口&#xff08;类似 iPhone 的 app&#…...

day011-权限管理专题

文章目录 1. 对比文件内容1.1 diff1.2 vimdiff 2. /etc/skel目录3. 权限基础4. 修改权限4.1 用数字权限修改4.2 用字母修改权限&#xff08;ugo&#xff09;4.3 修改文件所有者和用户组 5. 文件与目录权限6. permission denied 权限拒绝7. 特殊权限8. 特殊属性9. 思维导图 1. 对…...

ragflow报错:KeyError: ‘\n “序号“‘

环境&#xff1a; ragflowv 0.17.2 问题描述&#xff1a; ragflow报错&#xff1a;KeyError: ‘\n “序号”’ **1. 推荐表&#xff08;输出json格式&#xff09;** [{"},{},{"},{} ]raceback (most recent call last): May 08 20:06:09 VM-0-2-ubuntu ragflow-s…...

基于FPGA的PID控制器verilog实现,包含simulink对比模型

目录 1.课题概述 2.系统测试效果 3.核心程序与模型 4.系统原理简介 5.完整工程文件 1.课题概述 根据PID控制器的原理&#xff0c;设计FPGA的总体架构。通常包括误差计算模块、比例运算模块、积分运算模块、微分运算模块、加法器模块以及控制信号输出模块等。同时通过simul…...

互联网大厂Java面试实录:Spring Boot与微服务架构在电商场景中的应用解析

&#x1f4aa;&#x1f3fb; 1. Python基础专栏&#xff0c;基础知识一网打尽&#xff0c;9.9元买不了吃亏&#xff0c;买不了上当。 Python从入门到精通 &#x1f601; 2. 毕业设计专栏&#xff0c;毕业季咱们不慌忙&#xff0c;几百款毕业设计等你选。 ❤️ 3. Python爬虫专栏…...

前端开发实战:用React Hooks优化你的组件性能

问题背景 在前端开发中&#xff0c;React组件的性能优化是一个常见挑战。尤其是当组件逻辑复杂或数据频繁更新时&#xff0c;性能问题尤为突出。本文将介绍如何利用React Hooks&#xff08;如useMemo和useCallback&#xff09;来优化组件性能。 解决方案 useMemo&#xff1a;用…...

Kotlin 内联函数深度解析:从源码到实践优化

一、内联函数核心概念 1. 什么是内联函数&#xff1f; 内联函数通过 inline 关键字修饰&#xff0c;其核心思想是&#xff1a;在编译时将函数体直接插入到调用处&#xff0c;而非进行传统的函数调用。这意味着&#xff1a; 消除了函数调用的栈帧创建、参数传递等开销。对 La…...

模拟太阳系(C#编写的maui跨平台项目源码)

源码下载地址&#xff1a;https://download.csdn.net/download/wgxds/90789056 本资源为用C#编写的maui跨平台项目源码&#xff0c;使用Visual Studio 2022开发环境&#xff0c;基于.net8.0框架&#xff0c;生成的程序为“模拟太阳系运行”。经测试&#xff0c;生成的程序可运行…...

python中的继承和多态

Python中的继承 继承中的一些基础的定义 继承是面向对象编程的三大特性之一&#xff0c;它允许一个类&#xff08;子类&#xff09;继承另一个类&#xff08;父类&#xff09;的属性和方法&#xff0c;从而实现代码的复用&#xff08;继承的主要目的&#xff09;和扩展。父类…...

【计算机视觉】3DDFA_V2中表情与姿态解耦及多任务平衡机制深度解析

3DDFA_V2中表情与姿态解耦及多任务平衡机制深度解析 1. 表情与姿态解耦的技术实现1.1 参数化建模基础1.2 解耦的核心机制1.2.1 基向量正交化设计1.2.2 网络架构设计1.2.3 损失函数设计 1.3 实现代码解析 2. 多任务联合学习的权重平衡2.1 任务定义与损失函数2.2 动态权重平衡策略…...

vue访问后端接口,实现用户注册

文章目录 一、后端接口文档二、前端代码请求响应工具调用后端API接口页面函数绑定单击事件&#xff0c;调用/api/user.js中的函数 三、参考视频 一、后端接口文档 二、前端代码 请求响应工具 /src/utils/request.js //定制请求的实例//导入axios npm install axios import …...

MySQL 从入门到精通(五):索引深度解析 —— 性能优化的核心武器

目录 一、索引概述&#xff1a;数据库的 “目录” 1.1 什么是索引&#xff1f; 1.2 索引的性能验证&#xff1a;用事实说话 实验环境准备 无索引查询耗时 有索引查询耗时 索引的 “空间换时间” 特性 二、索引的创建&#xff1a;三种核心方式 2.1 方式 1&#xff1a;C…...

湖北理元理律师事务所:债务优化如何实现还款与生活的平衡?

债务压力往往让债务人陷入“还款还是生存”的两难选择。湖北理元理律师事务所通过案例实践发现&#xff0c;科学规划的核心在于平衡法律义务与基本生活保障&#xff0c;而非单纯追求债务缩减。本文结合实务经验&#xff0c;解析债务优化的可行路径。 刚性需求优先&#xff1a;…...

Day21 奇异值分解(SVD)全面解析

一、奇异值分解概述 奇异值分解是线性代数中一个重要的矩阵分解方法&#xff0c;对于任何矩阵&#xff0c;无论是结构化数据转化成的“样本 * 特征”矩阵&#xff0c;还是天然以矩阵形式存在的图像数据&#xff0c;都能进行等价的奇异值分解&#xff08;SVD&#xff09;。 二…...

【vue】vuex实现组件间数据共享 vuex模块化编码 网络请求

目录 一、vuex实现组件间数据共享 二、 vuex模块化编码 三、网络请求 模块化命名空间小结: 总结不易~ 本章节对我有很大的收获&#xff0c; 希望对你也是&#xff01;&#xff01;&#xff01; 本节素材已上传Gitee&#xff1a;yihaohhh/我爱Vue - Gitee.comhttps://gitee.…...

红黑树删除的实现与四种情况的证明

&#x1f9ed; 学习重点 删除节点的三种情况红黑树如何恢复性质四种修复情况完整可运行的 C 实现 一、红黑树删除的基础理解 红黑树删除比插入复杂得多&#xff0c;因为&#xff1a; 删除的是黑节点可能会破坏“从根到叶子黑节点数相等”的性质。删除红节点无需修复&#xf…...

FHE与后量子密码学

1. 引言 近年来&#xff0c;关于 后量子密码学&#xff08;PQC, Post-Quantum Cryptography&#xff09; 的讨论愈发热烈。这是因为安全专家担心&#xff0c;一旦有人成功研发出量子计算机&#xff0c;会发生什么可怕的事情。由于 Shor 算法的存在&#xff0c;量子计算机将能够…...

使用FastAPI和React以及MongoDB构建全栈Web应用04 MongoDB快速入门

一、NoSQL 概述 1.1 了解关系数据库的局限性 Before diving into NoSQL, it’s essential to understand the challenges posed by traditional Relational Database Management Systems (RDBMS). While RDBMS have been the cornerstone of data management for decades, th…...

C++:this指针

class date { public:void f(int i){} } 以上是我们定义的一个简单的类&#xff0c;这个类里面含有一个简单的成员函数&#xff0c;成员函数看似只有一个参数&#xff0c;实际上是两个参数&#xff0c;除了参数i以外&#xff0c;还有一个指向调用该函数的对象的指针——this指…...

如何在postman使用时间戳

1. 使用 Pre-request Script 动态转换​ 在发送请求前&#xff0c;将日期字符串转为时间戳并存储为环境变量/全局变量。 ​示例代码​ // 将日期字符串&#xff08;如 "2023-10-01"&#xff09;转为时间戳&#xff08;毫秒&#xff09; const dateString "2…...

OCP开闭原则

OCP&#xff0c;software entities(modules,classes,functions,etc.)should be openfor extension, but closed for modification. 软件实体&#xff08;模块、类和方法等&#xff09;应该对扩展开发&#xff0c;对修改关闭。 OCP特点 提高可扩展性&#xff1a;新功能通过添…...

计算机网络:什么是Mesh组网以及都有哪些设备支持Mesh组网?

Mesh组网技术详解与实现工具推荐 Mesh组网是一种通过多个节点路由器协同工作,形成覆盖全屋的无线网络的技术。它通过动态路径调整、无缝漫游和自愈能力,解决传统单一路由器覆盖不足的问题,尤其适合大户型、多层住宅或复杂户型环境。以下是Mesh组网的核心原理、实现方式及推…...