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

【SpringBoot3】面向切面 AspectJ AOP 使用详解

文章目录

    • 一、AspectJ介绍
    • 二、简单使用步骤
      • 1、引入依赖
      • 2、定义一个Aspect
      • 3、开启AOP支持
    • 三、AOP 核心概念
    • 四、切点(Pointcut)
      • 1. execution
      • 2. within
      • 3. this & target
      • 4. args & @args
      • 5. @within & @target & @annotation
    • 五、通知(Advice)
      • 1. @Before 示例:日志记录
      • 2. @AfterReturning 示例:处理返回值
      • 3. @AfterThrowing 示例:异常处理
      • 4. @After 示例:资源释放
      • 5. @Around 示例:方法执行前后处理
      • 6. 参数相关 Advice 示例:验证参数
      • 7. 注解相关 Advice 示例:基于注解的日志记录
      • 8. @AfterThrowing 异常示例:特定异常处理
      • 9. @Pointcut 单独定义切点,其他Advice直接引用
    • 参考

一、AspectJ介绍

AspectJ是一个面向切面的框架,它扩展了Java语言,并定义了AOP(面向切面编程)语法。

AspectJ支持数据埋点、日志记录、性能统计、安全控制、事务处理、异常处理等多种横切关注点。通过AspectJ,开发者可以更加直观地定义和理解代码的行为,减少对业务逻辑的干扰。

  • 特点
  1. 模块化:AspectJ允许开发者将横切关注点以模块化的方式进行管理和重用,提高了代码的可维护性和可读性。
  2. 声明式编程:AspectJ使用注解或XML配置的方式来声明横切关注点,减少了重复的样板代码。
  3. 细粒度控制:AspectJ提供了丰富的切点表达式语言,可以精确地选择需要横切的连接点,实现对代码的细粒度控制。
  4. 跨模块切面:AspectJ可以在不同的模块之间进行切面的织入,使得横切关注点可以跨越多个模块进行统一管理。
  5. 与Java语言的兼容性:AspectJ是基于Java语言的扩展,与Java语法完全兼容,可以无缝地与现有的Java代码进行集成和使用。

二、简单使用步骤

1、引入依赖

<dependency>  <groupId>org.springframework.boot</groupId>  <artifactId>spring-boot-starter-aop</artifactId>  
</dependency>

2、定义一个Aspect

import org.aspectj.lang.annotation.Aspect;  
import org.aspectj.lang.annotation.Before;  
import org.springframework.stereotype.Component;  @Aspect  
@Component  
public class MyAspect {  // 定义切点,表示对哪些类的哪些方法进行拦截  // execution表达式指定了方法签名  // 第一个"*"表示任意返回类型,第二个"*"表示类中的任意方法  // ".."表示任意参数  // 这里的例子表示拦截UserService中的所有方法  @Before("execution(* com.example.service.UserService.*(..))")  public void beforeAdvice() {  System.out.println("Before method: Do something before.");  }  
}

3、开启AOP支持

在Spring Boot应用中,通常通过@SpringBootApplication注解启动应用,该注解包含了@EnableAspectJAutoProxy,因此默认情况下Spring Boot应用是支持AOP的。

如果你的应用配置较为特殊,确保你的配置类上(通常是主类)添加了@EnableAspectJAutoProxy注解以开启对AspectJ自动代理的支持。

三、AOP 核心概念

  • 切面(Aspect):AOP 的模块单元,封装了切点、通知以及类型间声明。在Spring AOP中,切面通过使用常规类或使用@Aspect注解的常规类(@AspectJ风格)来实现。

  • 连接点(Join point):程序流中指定的一点,如方法调用、方法执行、字段访问等。

  • 通知(Advice):切面在特定连接点上采取的动作。通知包括“环绕”、“前置”和“后置”通知。许多AOP框架,包括Spring,都将通知建模为拦截器,并在连接点周围维护一个拦截器链。

  • 切点(Pointcut):匹配连接点的谓词。通知与切点表达式相关联。Spring默认使用AspectJ切点表达式语言。

  • 引入(Introduction):代表一个类型声明额外的方法或字段。Spring AOP允许您向任何被通知的对象引入新接口(以及相应的实现)。例如,您可以使用引入来使bean实现IsModified接口,以简化缓存。

  • 目标对象(Target object):被一个或多个切面通知的对象。也称为“被通知对象”。由于Spring AOP是通过使用运行时代理实现的,因此这个对象总是一个代理对象。

  • AOP代理(AOP proxy):由AOP框架创建的对象,用于实现切面契约(如通知方法的执行等)。在Spring框架中,AOP代理是JDK动态代理或CGLIB代理。

  • 织入(Weaving):将切面与其他应用类型或对象链接起来以创建被通知对象。这可以在编译时(例如,使用AspectJ编译器)、加载时或运行时完成。Spring AOP像其他纯Java AOP框架一样,在运行时执行织入。

Spring AOP包括以下类型的通知:

  • 前置通知(Before advice):在连接点之前运行的通知,但它没有能力阻止执行程序执行(除非它抛出异常)。
  • 后置返回通知(After returning advice):在连接点正常完成后运行的通知(例如,如果方法返回而没有抛出异常)。
  • 后置异常通知(After throwing advice):如果方法通过抛出异常退出时运行的通知。
  • 后置(最终)通知(After (finally) advice):无论连接点以何种方式退出(正常或异常返回)都要运行的通知。
  • 环绕通知(Around advice):围绕连接点(如方法调用)的通知。这是最强大的通知类型。环绕通知可以在方法调用之前和之后执行自定义行为。它还负责选择是否继续到连接点,或者通过返回自己的返回值或抛出异常来短路被通知方法的执行。

环绕通知是最通用的通知类型。由于Spring AOP像AspectJ一样提供了全范围的通知类型,因此建议您使用能够实现所需行为的最不强大的通知类型。例如,如果您只需要使用方法的返回值更新缓存,那么实现后置返回通知比实现环绕通知更好,尽管环绕通知也可以完成同样的事情。使用最具体的通知类型提供了更简单的编程模型,并且减少了出错的可能性。例如,您不需要在用于环绕通知的JoinPoint上调用proceed()方法,因此您不会忘记调用它。

所有通知参数都是静态类型的,因此您可以使用适当类型的通知参数(例如,方法执行的返回值的类型)而不是Object数组。

四、切点(Pointcut)

Spring AOP支持使用AspectJ切入点表达式来指定切点。这些表达式可以非常灵活地定义需要拦截的方法集合。以下是一些常用的切点指示符和示例:

1. execution

execution是最常用的切入点指示符,用于匹配方法执行的连接点。其语法结构如下:

execution(modifiers-pattern? ret-type-pattern declaring-type-pattern? name-pattern(param-pattern) throws-pattern?)
  • modifiers-pattern(可选):方法的修饰符,如publicprivate等。
  • ret-type-pattern:方法的返回类型,*表示任意类型。
  • declaring-type-pattern(可选):声明方法的类,支持使用..来表示包及其子包下的所有类。
  • name-pattern:方法的名称。
  • param-pattern:方法的参数列表,..表示任意数量和类型的参数,*表示任意类型的一个参数,(*, String)表示第一个参数是任意类型,第二个参数是String类型。
  • throws-pattern(可选):方法抛出的异常类型。

示例

  • 匹配任意类的任意方法:execution(* *(..))
  • 匹配com.example.service包下所有类的所有方法:execution(* com.example.service..*.*(..))
  • 匹配MyService类中的doSomething方法:execution(* com.example.service.MyService.doSomething(..))
  • 匹配任意类的save方法,且方法参数为java.lang.String类型:execution(* *.save(java.lang.String))

2. within

within用于匹配指定类型内的方法执行,包括指定的接口、类或包。

示例

  • 匹配com.example.dao包下的所有类的所有方法:within(com.example.dao.*)
  • 匹配com.example.dao包及其子包中所有类中的所有方法:within(com.example.dao..*)
  • 匹配实现了UserService接口的类的所有方法:within(com.example.service.UserService+)

3. this & target

  • this:用于匹配当前AOP代理对象类型的执行方法,注意是AOP代理对象的类型匹配。
  • target:用于匹配当前目标对象类型的执行方法,注意是目标对象的类型匹配。

示例

  • 匹配当前AOP代理对象类型为MyService的所有方法执行:this(com.example.service.MyService)
  • 匹配当前目标对象类型为MyService的所有方法执行:target(com.example.service.MyService)

4. args & @args

  • args:用于匹配当前执行的方法传入的参数为指定类型的执行方法,参数类型列表中的参数必须是类型全限定名,通配符不支持。
  • @args:匹配方法传入的参数所属类上拥有指定的注解的情况。

示例

  • 匹配方法参数为String类型的方法:args(java.lang.String)
  • 匹配方法参数类型上拥有@MyAnnotation注解的方法调用:@args(com.example.MyAnnotation)

5. @within & @target & @annotation

  • @within:匹配类级别上应用了特定注解的类中的所有方法。
  • @target:匹配运行时目标对象(代理对象)的类型上应用了特定注解的方法(在Spring AOP中常用于代理对象的切点定义)。
  • @annotation:匹配方法级别上应用了特定注解的方法。

示例

  • 匹配所有类上具体指定了@MyAnnotation注解的类内的所有方法:@within(com.example.MyAnnotation)
  • 匹配当前目标对象类型持有@MyAnnotation注解的方法:@target(com.example.MyAnnotation)
  • 匹配所有拥有@MyAnnotation注解的外部调用方法:@annotation(com.example.MyAnnotation)

五、通知(Advice)

Spring AOP提供了多种类型的Advice,包括:

  1. @Before:在目标方法执行之前执行。
  2. @AfterReturning:在目标方法正常执行完成后执行。
  3. @AfterThrowing:在目标方法抛出异常时执行。
  4. @After:无论目标方法执行结果如何,最终都会执行。
  5. @Around:在目标方法执行前后执行,并可以控制目标方法的执行。

每种类型的Advice都需要与切入点表达式结合使用,以确定其应用的范围。

1. @Before 示例:日志记录

在方法执行之前记录日志,常用于跟踪方法的调用。

@Aspect
@Component
public class LoggingAspect {// 定义切点,表示对哪些类的哪些方法进行拦截// execution表达式指定了方法签名// 第一个"*"表示任意返回类型,第二个"*"表示类中的任意方法// ".."表示任意参数// 这里的例子表示拦截UserService中的所有方法@Before("execution(* com.example.service.UserService.*(..))")public void logBeforeMethodExecution(JoinPoint joinPoint) {System.out.println("Before executing method: " + joinPoint.getSignature().getName());}// 定义切点,表示对哪些类的哪些方法进行拦截// execution表达式指定了方法签名// 第一个"*"表示任意返回类型,第二个"*"表示类中的任意方法// 第一次".."表示包及其子包下的所有类// 第二次".."表示任意参数@Before("execution(* com.example..*(..))")public void logBeforeMethodExecutionAll(JoinPoint joinPoint) {System.out.println("Before executing method all: " + joinPoint.getSignature().getName());}
}

2. @AfterReturning 示例:处理返回值

在方法成功执行并返回后,对返回值进行处理或记录。

@Aspect
@Component
public class ReturnValueAspect {@AfterReturning(pointcut = "execution(* com.example.service.UserService.getUserById(..))", returning = "result")public void logReturnValue(Object result) {System.out.println("Method returned: " + result);}
}

3. @AfterThrowing 示例:异常处理

在方法抛出异常时捕获异常,并进行相应的处理或记录。

@Aspect
@Component
public class ExceptionHandlingAspect {@AfterThrowing(pointcut = "execution(* com.example.service.*.*(..))", throwing = "ex")public void handleException(Exception ex) {System.err.println("An exception occurred: " + ex.getMessage());}
}

4. @After 示例:资源释放

无论方法执行结果如何,都会在方法执行后执行,常用于资源释放。

@Aspect
@Component
public class ResourceReleaseAspect {@After("execution(* com.example.service.*.*(..))")public void releaseResources() {System.out.println("Resources are released after method execution");}
}

5. @Around 示例:方法执行前后处理

在方法执行前后都进行处理,可以控制方法的执行或添加额外的逻辑。

@Aspect
@Component
public class AroundAdviceAspect {@Around("execution(* com.example.service.UserService.*(..))")public Object logAroundMethodExecution(ProceedingJoinPoint joinPoint) throws Throwable {System.out.println("Before executing method: " + joinPoint.getSignature().getName());Object result = joinPoint.proceed(); // 执行目标方法System.out.println("After executing method: " + joinPoint.getSignature().getName());return result;}
}

6. 参数相关 Advice 示例:验证参数

在方法执行之前验证参数的有效性。

@Aspect
@Component
public class ParameterValidationAspect {@Before("execution(* com.example.service.UserService.addUser(com.example.model.User)) && args(user)")public void validateUserParameter(User user) {if (user == null || user.getName() == null || user.getName().isEmpty()) {throw new IllegalArgumentException("User parameter is invalid");}}
}

7. 注解相关 Advice 示例:基于注解的日志记录

根据方法上的注解来决定是否进行日志记录。可以使用注解全限定名或使用参数中的注解参数名。

@Retention(RetentionPolicy.RUNTIME)
@Target(ElementType.METHOD)
public @interface Loggable {
}@Aspect
@Component
public class AnnotationDrivenLoggingAspect {@Before("@annotation(com.example.aspect.Loggable)")public void logMethodWithAnnotation(JoinPoint joinPoint) {System.out.println("Executing loggable method: " + joinPoint.getSignature().getName());}// 使用参数中的注解@Before("@annotation(loggable)")public void logMethodWithAnnotationArg(JoinPoint joinPoint, Loggable loggable) {System.out.println("Executing loggable method: " + joinPoint.getSignature().getName());}
}// 在需要日志记录的方法上使用@Loggable注解
@Service
public class MyService {@Loggablepublic void myLoggableMethod() {// ...}
}

8. @AfterThrowing 异常示例:特定异常处理

仅当方法抛出特定类型的异常时进行处理。

@Aspect
@Component
public class SpecificExceptionHandlingAspect {@AfterThrowing(pointcut = "execution(* com.example.service.*.*(..))", throwing = "ex")public void handleSpecificException(CustomException ex) {System.err.println("A CustomException occurred: " + ex.getMessage());}
}// 自定义异常类
public class CustomException extends RuntimeException {public CustomException(String message) {super(message);}
}

9. @Pointcut 单独定义切点,其他Advice直接引用

@Aspect
@Component
public class LoggingAspect {// 定义切点,表示对哪些类的哪些方法进行拦截// execution表达式指定了方法签名// 第一个"*"表示任意返回类型,第二个"*"表示类中的任意方法// ".."表示任意参数// 这里的例子表示拦截UserService中的所有方法@Pointcut("execution(* com.example.service.UserService.*(..))")private void service() {}@Before("service()")public void logBeforeMethodExecution(JoinPoint joinPoint) {System.out.println("Before executing method: " + joinPoint.getSignature().getName());}}

参考

  • https://docs.spring.io/spring-framework/reference/core/aop.html

相关文章:

【SpringBoot3】面向切面 AspectJ AOP 使用详解

文章目录 一、AspectJ介绍二、简单使用步骤1、引入依赖2、定义一个Aspect3、开启AOP支持 三、AOP 核心概念四、切点&#xff08;Pointcut&#xff09;1. execution2. within3. this & target4. args & args5. within & target & annotation 五、通知&#xff0…...

wav怎么转mp3格式?给你推荐几种音频格式转换方法

wav怎么转mp3格式&#xff1f;将wav文件转换为MP3格式是一个常见的操作&#xff0c;尤其适用于需要节省存储空间或确保文件兼容性的场景。wav文件保存了音频的所有原始数据&#xff0c;这使得它们的文件体积往往非常庞大。相比之下&#xff0c;MP3格式通过有损压缩技术显著减小…...

Redis的AOF持久化、重写机制、RDB持久化、混合持久化

1、AOF持久化 1.1.AOF持久化大致过程 概括&#xff1a;命令追加&#xff08;append&#xff09;、文件写入、文件同步&#xff08;sync&#xff09; Redis 每执行一条写操作命令&#xff0c;就把该命令以追加的方式写入到一个文件里&#xff0c;然后重启 Redis 的时候&#…...

Dom4j使用xpath查询xml文

Dom4j使用xpath查询带有命名空间的xml文件 方式1 忽略命名空间 DocumentFactory factory DocumentFactory.getInstance(); SAXReader reader new SAXReader(factory); Document document reader.read(xmlFilePath); Element rootElement document.getRootElement(); Nod…...

国家专精特新小巨人企业指标解析与扶持领域

一、什么是国家专精特新小巨人 &#xff08;一&#xff09;概念与定义 专精特新“小巨人”企业是指那些在细分市场中具有专业化、精细化、特色化和新颖化特征的中小企业中的佼佼者。这些企业在创新能力强、市场占有率高、掌握关键核心技术以及质量效益方面表现突出&#xff0…...

进程的属性

tips&#xff1a; task_struct就是linux下的PCB 操作系统不相信任何外部用户&#xff0c;而是只提供窗口&#xff0c;不可能直接与用户打交道&#xff0c;而是通过操作系统 tast_struct用来描述所有进程&#xff0c;用来管理 &#xff1b; 和 && 可以同时跑两个命令 进…...

Git 中的refs

在 Git 中&#xff0c;refs 是用来存储 Git 对象&#xff08;如提交、树、标签等&#xff09;的引用。每个 ref 都是一个指针&#xff0c;指向一个特定的 Git 对象。以下是 Git 中几种常见的 refs 及其含义&#xff1a; 1. refs/heads/ 表示&#xff1a;本地分支。 用途&…...

408算法题leetcode--第六天

58. 最后一个单词的长度 58. 最后一个单词的长度思路&#xff1a;反向遍历时间&#xff1a;O(n)&#xff1b;空间&#xff1a;O(1) class Solution { public:int lengthOfLastWord(string s) {int id s.size() - 1;while(s[id] ){--id;}int ret 0;while(id > 0 &&…...

ubuntu64位系统无法运行32位程序的解决办法

在 64 位的 Ubuntu 系统上运行 32 位程序时&#xff0c;如果出现问题&#xff0c;可能是由于缺少 32 位库支持。以下步骤可以帮助你解决这一问题&#xff1a; 1. 启用 32 位架构 首先&#xff0c;确保系统支持 32 位架构。你可以通过以下命令添加 32 位架构支持&#xff1a; …...

深入理解Go语言中的并发封闭与for-select循环模式

在现代编程中,并发已经成为提高程序性能和响应能力的关键手段。然而,在并发环境下,如何安全地访问和操作共享数据却是一大挑战。本文将深入探讨Go语言中的**封闭(confinement)**技术,以及常见的for-select循环模式,帮助您编写更高效、更安全的并发代码。 并发编程中的安…...

Java学习Day42:骑龙救!(springMVC)

springMVC与sevlet都是对应表现层web的&#xff0c;但是越复杂的项目使用SpringMVC越方便 基于Java实现MVC模型的轻量级web框架 目标&#xff1a; 小案例&#xff1a; 1.导入依赖 spring-context: 提供 Spring 框架的核心功能&#xff0c;如依赖注入、事件发布和其他应用上…...

原型模式详细介绍和代码实现

&#x1f3af; 设计模式专栏&#xff0c;持续更新中&#xff0c; 欢迎订阅&#xff1a;JAVA实现设计模式 &#x1f6e0;️ 希望小伙伴们一键三连&#xff0c;有问题私信都会回复&#xff0c;或者在评论区直接发言 Java实现原型模式 介绍: 原型模式&#xff08;Prototype Patte…...

ArcGIS Pro SDK (十三)地图创作 5 图层样式

ArcGIS Pro SDK (十三)地图创作 5 图层样式 文章目录 ArcGIS Pro SDK (十三)地图创作 5 图层样式1 风格管理1.1 如何按名称获取项目中的样式1.2 如何创建新样式1.3 如何向项目添加样式1.4 如何从项目中删除样式1.5 如何将样式项添加到样式1.6 如何从样式中删除样式项1.7 如…...

【Python报错已解决】 Requests.exceptions.ProxyError: HTTPSConnectionPool

&#x1f3ac; 鸽芷咕&#xff1a;个人主页 &#x1f525; 个人专栏: 《C干货基地》《粉丝福利》 ⛺️生活的理想&#xff0c;就是为了理想的生活! 专栏介绍 在软件开发和日常使用中&#xff0c;BUG是不可避免的。本专栏致力于为广大开发者和技术爱好者提供一个关于BUG解决的经…...

现在量化中普遍使用QMT和PTrade?哪家可以同时提供QMT/PTrade?

QMT的特点 全面的功能集成&#xff1a; QMT集成了行情显示、策略研究、交易执行和风控管理于一体&#xff0c;为投资者提供了一站式的量化交易解决方案。 高效的交易执行能力&#xff1a; 通过全内存交易实现低延迟的交易执行&#xff0c;单笔延时小于1ms&#xff0c;确保了交易…...

【计算机网络】UDP 协议详解及其网络编程应用

文章目录 一、引言二、UDP1、UDP的协议格式2、UDP 报文的解包和分用3、UDP面向数据报的特点 三、UDP输入输出四、UDP网络编程 一、引言 UDP&#xff08;User Datagram Protocol&#xff0c;用户数据报协议&#xff09;是一种网络通信协议&#xff0c;它属于传输层的协议。是一…...

基于ESP32S3的链接大语言模型对话模块

本实物模块从实物外观、模块组成、API申请及功能说明四部分来介绍这款基于ESP32S3的大语言模型对话模块。 1、实物外观 2、模块介绍 本硬件平台主要由三个模块组成&#xff0c;包括MAX9814录音模块、MAX98357音频功放模块和ESP32S3模块。如下图所示。 MAX9814录音模块&#…...

Cpp输出多字符常量警告

Cpp输出多字符常量警告 Cpp中用单引号(single quotes)表示单个字符(single character)&#xff0c;例如a&#xff0c;$&#xff0c;用双引号(double quotes)表示字符串文本(text)&#xff0c;例如"Hello World! " 当在一个单引号里面存在多个字符时&#xff0c;Cpp…...

Maven从入门到精通(二)

一、什么是pom.xml pom.xml是Maven项目的核心配置文件&#xff0c;它是 项目对象模型 - Project Object Model&#xff08;POM&#xff09;的缩写。POM定义了项目的所有属性&#xff0c;包括项目的名称、版本、依赖关系、构建配置等。使用pom.xml&#xff0c;我们可以轻松地管…...

【Kubernetes】常见面试题汇总(二十四)

目录 71.假设一家公司想要修改它的部署方法&#xff0c;并希望建立一个更具可扩展性和响应性的平台。您如何看待这家公司能够实现这一目标以满足客户需求&#xff1f; 72.考虑一家拥有非常分散的系统的跨国公司&#xff0c;期待解决整体代码库问题。您认为公司如何解决他们的问…...

最低成本的游戏串流方案分享 如何自己打造云电脑?

今天教大家如何最低成本实现串流 出门在外也可以随时随地游玩端游大作 硬件准备&#xff1a;一台电脑 手机/平板一台 软件&#xff1a;Gameviewer远程 为啥不用moonlight等其他软件呢 因为设置公网穿透等复杂操作对小白来说不太友好 而GameViewer从安装到使用仅需一键 对比同类…...

python运行时错误:找不到fbgemm.dll

python运行时错误&#xff1a;找不到fbgemm.dll 报错&#xff1a; OSError: [WinError 126] 找不到指定的模块。 Error loading "D:\program\py\312\Lib\site-packages\torch\lib\fbgemm.dll" or one of its dependencies. 原因是Windows下缺失&#xff1a;libomp140…...

给虚拟机linux系统安装交叉编译工具链

我们在电脑上写的代码编译生成的是X86架构的二进制文件&#xff0c;只能在X86平台上运行&#xff0c;而开发板是ARM架构因此需要安装交叉编译链工具&#xff0c;这样在电脑上写的代码交叉编译之后生成的是ARM架构的二进制文件。 绿色的字眼是与本文无关的只是这样有助于我们的…...

Redhat 7,8系(复刻系列) 一键部署Oracle21c-xe rpm

Oracle21c-xe前言 无论您是开发人员、DBA、数据科学家、教育工作者,还是仅仅对数据库感兴趣,Oracle Database Express Edition (XE) 都是理想的入门方式。它是全球企业可依赖的强大的 Oracle Database,提供简单的下载、易于使用和功能齐全的体验。您可以在任何环境中使用该…...

Web植物管理系统-下位机部分

本节主要展示上位机部分&#xff0c;采用BSP编程&#xff0c;不附带BSP中各个头文件的说明&#xff0c;仅仅是对main逻辑进行解释 main.c 上下位机通信 通过串口通信&#xff0c;有两位数据验证头&#xff08;verify数组中保存对应的数据头 0xAA55) 通信格式 上位发送11字节…...

leetcode169. 多数元素

给定一个大小为 n 的数组 nums &#xff0c;返回其中的多数元素。多数元素是指在数组中出现次数 大于 ⌊ n/2 ⌋ 的元素。 你可以假设数组是非空的&#xff0c;并且给定的数组总是存在多数元素。 示例 1&#xff1a; 输入&#xff1a;nums [3,2,3] 输出&#xff1a;3 示例…...

从大脑图谱/ROI中提取BOLD信号

动机 在功能连接&#xff08;Functional Connectivity&#xff0c;FC&#xff09;构建过程中&#xff0c;由于FC中元素数目是节点数目的平方关系&#xff0c;所以在计算FC之前进行数据降维是一个常见的选择。 一般会将体素级/顶点级BOLD信号&#xff08;在2mm的图像分辨率下大脑…...

Java-数据结构-优先级队列(堆)-(一) (;´д`)ゞ

文本目录&#xff1a; ❄️一、优先级队列&#xff1a; ➷ 1、概念&#xff1a; ❄️二、优先级队列的模拟实现&#xff1a; ➷ 1、堆的概念&#xff1a; ➷ 2、堆的性质&#xff1a; ➷ 3、堆的创建&#xff1a; ▶ 向下调整&#xff1a; ➷ 4、堆的插入和删除&#xff1a; …...

工厂模式(二):工厂方法模式

一、概念 工厂方法模式&#xff08;Factory Method&#xff09;&#xff0c;定义一个用于创建对象的接口&#xff0c;让子类决定实例化哪一个类。工厂方法使一个类的实例化延迟到其子类。从而使得系统更加灵活。客户端可以通过调用工厂方法来创建所需的产品&#xff0c;而不必…...

【洛谷】P11036 【MX-X3-T3】「RiOI-4」GCD 与 LCM 问题 的题解

【洛谷】P11036 【MX-X3-T3】「RiOI-4」GCD 与 LCM 问题 的题解 题目传送门 题解 神奇构造题qaq 简化一下下题目&#xff0c;就是要求 a b c d gcd ⁡ ( a , b ) lcm ⁡ ( c , d ) a b c d \gcd(a, b) \operatorname{lcm}(c,d) abcdgcd(a,b)lcm(c,d) 分类讨论 …...