Spring AOP:面向切面编程 详解代理模式

article/2025/6/8 0:20:33

文章目录

  • AOP介绍
  • 什么是Spring AOP?
    • 快速入门SpringAop
      • 引入依赖
      • Aop的优点
    • Spring Aop 的核心概念
      • 切点(Pointcut)
      • 连接点、
      • 通知
      • 切面
      • 通知类型
      • @PointCut注解
      • 切面优先级@Order
      • 切点表达式
        • execution
        • within
        • this
        • target
        • args
        • @annotation
          • 自定义注解
  • Spring AOP原理
    • 代理模式(委托模式)
      • 静态代理
      • 动态代理
        • JDK动态代理类实现步骤
      • CGLIB动态代理
        • CGLIB 动态代理类实现步骤
  • 总结

AOP介绍

AOP(Aspect Oriented Programming 面向切面编程)

什么是面向切面编程呢?切面就是指某⼀类特定问题

所以AOP也可以理解为面向特定方法编程.

什么是面向特定方法编程?

比如说上篇博客中写到的登录校验拦截器,就是对于某一问题的集中处理, AOP是一种思想,拦截器是AOP的一种实现,统一处理异常,统一返回数据格式,也是 AOP的一种实现

Spring框架实现了这种思想,提供了拦截器相关技术的接口

简单来说: AOP是⼀种思想,是对某⼀类事情的集中处理

什么是Spring AOP?

SpringAop是Spring框架提供的一种面向切面编程的技术。也就是说对 AOP思想的一种实现。它通过将横切关注点(例如日志记录、事务管理、安全性检查等)从主业务逻辑代码中分离出来,以模块化的方式实现对这些关注点的管理和重用。

在Spring AOP中,切面(Aspect)是一个模块化的关注点,它可以跨越多个对象,例如日志记录、事务管理等。切面通过定义切点(Pointcut)和增强(Advice)来介入目标对象的方法执行过程。

在日常的代码编写中,有以下例子:

在这里插入图片描述

假如,我们要对这里面的接口实现改良,让执行时间减少,这里就定位到某些业务代码逻辑等改进,但是不是所有都是不好的,所以我们需要进行测试每个接口,业务执行的时间,那么就有如下的代码,来进行时间检测:

在这里插入图片描述
此时的方法固然是可以的,但是那么多接口都需要这样写吗??答案是不可能的,此时这就是一类特定的问题,此时我们就可以使用AOP思想来进行解决

AOP就可以做到在不改动这些原始方法的基础上, 针对特定的方法进行功能的增强.
AOP的作用:在程序运行期间在不修改源代码的基础上对已有方法进行增强(无侵入性: 解耦)

快速入门SpringAop

引入依赖

Maven依赖

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

编写Aop程序
记录Controller中每个方法的执行时间

@Slf4j
@Component
@Aspect
public class TimeAspect {@Around("execution(* com.spring.aop.springaop.controller.*.*(..))")public Object readTime (ProceedingJoinPoint pjp) throws Throwable {//开始时间long begin=System.currentTimeMillis();//执行方法Object result=pjp.proceed();//结束时间long end=System.currentTimeMillis();//记录耗时时间log.info(pjp.getSignature()+"耗时:{}",end-begin+"ms");return result;}}

在这里插入图片描述
运行程序,观察日志
在这里插入图片描述
对于程序进行简单的讲解:

  1. @Aspect:标识这是⼀个切面类
  2. @Around:环绕通知,在目标方法的前后都会被执行.后面的表达式表示对哪些方法进行增强.
  3. ProceedingJoinPoint.proceed() 让原始方法执行

Aop的优点

提高代码复用率:通过将通用功能,如日志记录或权限检查,封装在独立的切面中,多个地方能够复用这些功能。(减少重复代码,提高开发效率)
解耦业务逻辑:AOP 使得关注点的逻辑与核心业务逻辑分离,降低了系统各部分之间的依赖性。(代码无侵入)
集中处理关注点:相关的代码可以集中于一个地方进行管理,简化了维护过程。(维护方便)

Spring Aop 的核心概念

切点(Pointcut)

也称为切入点
作用:提供⼀组规则告诉程序对哪些方法来进行功能增强.

execution(* com.spring.aop.springaop.controller..(…))是切点表达式

连接点、

满足切点表达式规则的方法,就是连接点,也就是可以被AOP控制的方法
即com.spring.aop.springaop.controller路径下的类路径下的方法

在这里插入图片描述

通知

通知就是具体要做的工作,指哪些重复的逻辑,也就是共性功能(最终体现为⼀个方法)
比如上述中记录耗时方法就是一个通知

切面

切面(Aspect)=切点(Pointcut)+通知(Advice)

通过切面就能够描述当前AOP程序需要针对于哪些方法,在什么时候什么样的操作
被@Aspect注解修饰的类,称为切面类

通知类型

通知类型解释
@Around环绕通知,在目标方法前,后都被执行
@Before前置通知,在目标方法前被执行
@After后置通知,在目标方法后被执行,无论是否有异常都会执行
@AfterReturning返回后通知,在目标方法后执行 ,有异常不会执行
@AfterThrowing异常后通知,在发生异常后执行

测试程序,加深了解

编写切面类

@Slf4j
@Aspect
@Component
public class AspectDemo1 {@Pointcut("execution(* com.spring.aop.springaop.controller.*.*(..))")public void pt(){}@Around("pt()")public Object recordTime(ProceedingJoinPoint pjp) throws Throwable {log.info("目标方法执行前....");//执行目标方法Object result = null;try {result = pjp.proceed();} catch (Exception e) {log.error("do Around throwing....");}//记录方法执行结束时间log.info("目标方法执行后....");return result;}@Before("pt()")public void doBefore() {log.info("doBefore....");}@After("pt()")public void doAfter(){log.info("doAfter....");}@AfterReturning("pt()")public void doAfterReturning(){log.info("doAfterReturnin....");}@AfterThrowing("pt()")public void doAfterThrowing(){log.info("doAfterThrowing....");}}

测试类

@Slf4j
@RequestMapping("/test")
@RestController
public class TestController {@RequestMapping("/t1")public Integer t1(){log.info("执行t1");int a = 10/0;return 1;}@MyAspect@RequestMapping("/t2")public Boolean t2(){log.info("执行t2");return true;}@RequestMapping("/t3")public String t3(){log.info("执行t3");return "t3";}
}

运行test/t1,除数异常
在这里插入图片描述
程序发生异常的情况下:

• @AfterReturning 标识的通知方法不会执行, @AfterThrowing 标识的通知方法执行了

• @Around 环绕通知中原始方法调用时有异常,通知中的环绕后的代码逻辑也不会在执行了

注意事项:
• @Around 环绕通知需要调用ProceedingJoinPoint.proceed() 来让原始方法执行,其他通知不需要考虑目标方法执行.

• @Around 环绕通知方法的返回值,必须指定为Object,来接收原始方法的返回值,否则原始方法执行完毕,是获取不到返回值的.

• 一个切面类可以有多个切点.

@PointCut注解

@Slf4j
@Aspect
@Component
public class AspectDemo1 {@Pointcut("execution(* com.spring.aop.springaop.controller.*.*(..))")public void pt(){}
}
  @Before("com.spring.aop.springaop.aspect.AspectDemo1.pt()")public void doBefore() {log.info("AspectDemo2 doBefore....");}

当切点定义使用private修饰时,仅能在当前切面类中使用

当其他切面类也要使用当前切点定义时,就需要把private改为public.引用方式为:全限定类名.方法名()

切面优先级@Order

@Slf4j
@Aspect
@Order(1)
@Component
public class AspectDemo4 {@Before("execution(* com.spring.aop.springaop.controller.*.*(..))")public void doBefore() {log.info("AspectDemo4 doBefore....");}@After("execution(* com.spring.aop.springaop.controller.*.*(..))")public void doAfter(){log.info("AspectDemo4 doAfter....");}}
@Slf4j
@Aspect
@Order(3)
@Component
public class AspectDemo2 {//    @Before("execution(* com.spring.aop.springaop.controller.*.*(..))")@Before("com.spring.aop.springaop.aspect.AspectDemo1.pt()")public void doBefore() {log.info("AspectDemo2 doBefore....");}@After("execution(* com.spring.aop.springaop.controller.*.*(..))")public void doAfter(){log.info("AspectDemo2 doAfter....");}}
@Slf4j
@Order(5)
@Aspect
@Component
public class AspectDemo3 {@Before("execution(* com.spring.aop.springaop.controller.*.*(..))")public void doBefore() {log.info("AspectDemo3 doBefore....");}@After("execution(* com.spring.aop.springaop.controller.*.*(..))")public void doAfter(){log.info("AspectDemo3 doAfter....");}}

指定执行顺序在这里插入图片描述

通过观察运行结果可以发现:
@Order注解标识的切面类,执行顺序如下
• @Before 通知:数字越小先执行
• @After 通知:数字越大先执行

@Order 控制切面的优先级,先执行优先级较高的切面,再执行优先级较低的切面,最终执行目标方法.
在这里插入图片描述

切点表达式

execution

execution([可见性] 返回类型 [声明类型].方法名(参数)[异常]),其中[]内的是可选的,其他的还支持通配符的使用:

  1. *: 匹配所有

  2. … : 匹配多个包或多个参数

  3. +: 表示类及其子类

  4. 运算符:&&、||、!

within

是用来指定类型的,指定类型中的所有方法将被拦截是用来指定类型的,指定类型中的所有方法将被拦截

within(com.demo.service.impl.UserServiceImpl) 匹配UserServiceImpl类对应对象的所有方法调用,并且只能是UserServiceImpl对象,不能是它的子对象

within(com.demo…*)匹配com.demo包及其子包下面的所有类的所有方法的外部调用

this

SpringAOP是基于代理的,this就代表代理对象,语法是this(type),当生成的代理对象可以转化为type指定的类型时表示匹配。

this(com.demo.service.UserService)匹配生成的代理对象是UserService类型的所有方法的外部调用

target

SpringAOP是基于代理的,target表示被代理的目标对象,当被代理的目标对象可以转换为指定的类型时则表示匹配。

target(com.demo.service.IUserService) 匹配所有被代理的目标对象能够转化成IuserService类型的所有方法的外部调用。

args

args用来匹配方法参数

args() 匹配不带参数的方法

args(java.lang.String) 匹配方法参数是String类型的

args(…) 带任意参数的方法

args(java.lang.String,…) 匹配第一个参数是String类型的,其他参数任意

@annotation

带有相应标注的任意方法,比如@Transactional或其他自定义注解
在这里插入图片描述

自定义注解
@Target({ElementType.METHOD})
@Retention(RetentionPolicy.RUNTIME)
public @interface MyAspect {
}
public class MyAspectImpl {@Around("@annotation(com.spring.aop.springaop.aspect.MyAspect)")
//@Around("@annotation(org.springframework.web.bind.annotation.RequestMapping)")public Object recordTime(ProceedingJoinPoint pjp) {log.info("目标方法执行前....");//执行目标方法Object result = null;try {result = pjp.proceed();} catch (Throwable e) {log.error("do Around throwing....");}log.info("目标方法执行后....");return result;}
}

测试类

@Slf4j
@RequestMapping("/user")
@RestController
public class UserController {@MyAspect@RequestMapping("/u2")public String u2(){log.info("执行u2");return "u2";}
}

在这里插入图片描述

@Target 标识了 Annotation 所修饰的对象范围, 即该注解可以用在什么地方

常用取值:
ElementType.TYPE: 用于描述类、接口(包括注解类型)或enum声明
ElementType.METHOD: 描述方法
ElementType.PARAMETER: 描述参数
ElementType.TYPE_USE: 可以标注任意类型

@Retention 指Annotation被保留的时长短,标明注解的生命周期

@Retention 的取值有三种:

  1. RetentionPolicy.SOURCE:表示注解仅存在于源代码中,编译成字节码后会被丢弃.
    这意味着在运行时无法获取到该注解的信息,只能在编译时使用.比如 @SuppressWarnings 以及lombok提供的注解 @Data @Slf4j

  2. RetentionPolicy.CLASS:编译时注解.表示注解存在于源代码和字节码中,但在运行时会被丢弃.这意味着在编译时和字节码中可以通过反射获取到该注解的信息,但在实际运行时无法获
    取.通常用于⼀些框架和工具的注解.

  3. RetentionPolicy.RUNTIME:运行时注解.表示注解存在于源代码,字节码和运行时中.这意味着在编译时,字节码中和实际运行时都可以通过反射获取到该注解的信息.通常用于⼀些需要在运行时处理的注解,如Spring的@Controller @ResponseBody

Spring AOP原理

Spring AOP是基于动态代理实现的

代理模式(委托模式)

定义:为其他对象提供⼀种代理以控制对这个对象的访问.
作用:通过提供⼀个代理类,让我们在调用目标方法的时候,不再是直接对目标方法进行调用,而是是通过代理类间接调用.
在某些情况下,⼀个对象不适合或者不能直接引用另⼀个对象,而代理对象可以在客户端和目标对象之间起到中介的作用.

使用代理前:
在这里插入图片描述
使用代理后
在这里插入图片描述
生活中的代理:
房屋中介:卖方会把房屋授权给中介,由中介来代理看房,房屋咨询等服务.
艺人经纪人:广告商找艺人拍广告,需要经过经纪人,由经纪人来和艺人进行沟通.
代理模式中的主要角色

  1. Subject: 业务接口类.可以是抽象类或者接口(不⼀定有)
  2. RealSubject: 业务实现类.具体的业务执行,也就是被代理对象.
  3. Proxy: 代理类.RealSubject的代理.

比如房屋租赁
Subject:提前定义了房东做的事情,交给中介代理
RealSubject:房东
Proxy:中介

UML类图

在这里插入图片描述
代理模式可以在不修改被代理对象的基础上,通过扩展代理类,进行一些功能的附加与增强.

根据代理的创建时期,代理模式分为静态代理和动态代理.

• 静态代理: 由程序员创建代理类或特定工具自动生成源代码再对其编译,在程序运行前代理类的.class文件就已经存在了.

• 动态代理: 在程序运行时,运用反射机制动态创建而成.

静态代理

我们通过代码来加深理解.以房租租赁为例

定义接口(房东要中介做的事情,subject)

public interface HouseSubject {void rentHouse();
}

实现接口(房东出租房子)

public class RealHouseSubject implements HouseSubject{@Overridepublic void rentHouse() {System.out.println("我是房东, 我出租房子");}
}

代理(中介帮房东出租房子)

public class HouseProxy implements HouseSubject{//将被代理对象声明为成员变量private HouseSubject realHouseSubject;public HouseProxy(HouseSubject realHouseSubject) {this.realHouseSubject = realHouseSubject;}@Overridepublic void rentHouse() {System.out.println("我是中介, 我帮房东开始代理");realHouseSubject.rentHouse();System.out.println("我是中介, 我帮房东结束代理");}
}

使用

public class Main {public static void main(String[] args) {HouseSubject target = new RealHouseSubject();HouseSubject houseSubject = new HouseProxy(target);houseSubject.rentHouse();
}

在这里插入图片描述
从上述程序可以看出,虽然静态代理也完成了对目标对象的代理,但是由于代码都写死了,对目标对象的每个方法的增强都是手动完成的,非常不灵活.
所以日常开发几乎看不到静态代理的场景.

如果需要增加新的需求,又要修改和新增业务实现类和代理类

既然代理的流程是⼀样的,有没有⼀种办法,让他们通过⼀个代理类来实现呢?
这就需要动态代理技术了.

动态代理

Java也对动态代理进行了实现,并给我们提供了⼀些API,常见的实现方式有两种:

  1. JDK动态代理
  2. CGLIB动态代理

动态代理在我们日常开发中使用的相对较少,但是在框架中几乎是必用的⼀门技术.学会了动态代理之后,对于我们理解和学习各种框架的原理也非常有帮助.

JDK动态代理类实现步骤

定义JDK动态代理类

实现InvocationHandler类

public class JDKInvocation implements InvocationHandler {/*** 目标对象, 被代理对象*/private Object target;public JDKInvocation(Object target) {this.target = target;}//调用目标方法@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {//代理增强内容System.out.println("我开始代理...");//通过反射调用被代理类的方法Object invoke = method.invoke(target, args);//代理增强内容System.out.println("我结束代理");return invoke;}
}

创建代理对象并使用

public class Main {public static void main(String[] args) {HouseSubject target = new RealHouseSubject();//JDK动态代理//动态创建一个代理类创建⼀个代理类:通过被代理类、被代理实现的接⼝、⽅法调⽤处理器来创建 HouseSubject houseProxy = (HouseSubject) Proxy.newProxyInstance(target.getClass().getClassLoader(),new Class[]{HouseSubject.class}, new JDKInvocation(target));houseProxy.rentHouse();}
}

代码讲解:InvocationHandler接口是Java动态代理的关键接口之⼀,它定义了⼀个单一方法invoke() 用于处理被代理对象的方法调用

通过实现 InvocationHandler 接口,可以对被代理对象的方法进行功能增强.
Proxy 类中使用频率最高的方法是: newProxyInstance() ,这个方法主要用来生成⼀个代理对象

通过Proxy.newProxyInstance(ClassLoader loader,Class<?>[]
interfaces,InvocationHandler h) 方法创建代理对象

Loader: 类加载器,用于加载代理对象.

interfaces: 被代理类实现的⼀些接口(这个参数的定义,也决定了JDK动态代理只能代理实现了接口的⼀些类)

h: 实现了 InvocationHandler接口的对象

JDK动态代理有⼀个最致命的问题是其只能代理实现了接口的类.
有些场景下,我们的业务代码是直接实现的,并没有接口定义.为了解决这个问题,我们可以用CGLIB动态代理机制来解决.

CGLIB动态代理

CGLIB(Code Generation Library)是⼀个基于ASM的字节码生成库,它允许我们在运行时对字节码进行修改和动态生成.

CGLIB通过继承方式实现代理,很多知名的开源框架都使用到了CGLIB.

例如Spring中的Aop模块中: 如果目标对象实现了接口,则默认采用JDK动态代理,否则采用CGLIB动态代理.

CGLIB 动态代理类实现步骤
  1. 定义⼀个类(被代理类)
  2. 自定义MethodInterceptor 并重写intercept 方法, intercept 用于增强目标方法,和JDK动态代理中的invoke 方法类似
  3. 通过Enhancer类的create()创建代理类

添加依赖

<dependency><groupId>cglib</groupId><artifactId>cglib</artifactId><version>3.3.0</version>
</dependency>

实现MethodInterceptor接口

public class CGlibMethodInterceptor implements MethodInterceptor {//目标对象private Object target;public CGlibMethodInterceptor(Object target) {this.target = target;}@Overridepublic Object intercept(Object obj, Method method, Object[] args, MethodProxy proxy) throws Throwable {System.out.println("我开始代理");Object invoke = method.invoke(target, args);System.out.println("我结束代理");return invoke;}
}

创建代理类

public class Main {public static void main(String[] args) {HouseSubject target = new RealHouseSubject();HouseSubject houseProxy = (HouseSubject) Enhancer.create(target.getClass(), new CGlibMethodInterceptor(target));houseProxy.rentHouse();}
}

代码讲解:MethodInterceptor 和JDK动态代理中的 InvocationHandler 类似,它只定义了⼀个方法intercept() ,用于增强目标方法.

Enhancer.create()用来生成⼀个代理对象
public static Object create(Class type, Callback callback) {
}
参数说明:
type: 被代理类的类型(类或接口)
callback: 自定义方法拦截器 MethodInterceptor

总结

  1. AOP是⼀种思想,是对某⼀类事情的集中处理.Spring框架实现了AOP,称之为SpringAOP
  2. Spring AOP常见实现方式有两种:1.基于注解@Aspect来实现2.基于自定义注解来实现,还有一些更原始的方式,比如基于xml配置的方式,但比较少见
  3. Spring AOP 是基于动态代理实现的,有两种方式:1.基本JDK动态代理实现 2.基于CGLIB动态代理实现.运行使用哪种方式与项目配置和代理的对象有关.

http://www.hkcw.cn/article/oqoWPHfKdU.shtml

相关文章

Ubuntu20.04用root(管理员身份)启动vscode

1.终端输入 code --user-data-dir~/.vscode --no-sandbox .

第7章 :面向对象

一、面向对象 面向过程&#xff1a;关心的是我该怎么做&#xff0c;一步步去实现这个功能 面向对象&#xff1a;关心的是我该让谁去做&#xff0c;去调用对象的操作来实现这个功能 面向对象 对象&#xff08;Object&#xff09; 分类&#xff08;Classification&#xff09; …

嵌入式linux八股文

1.指针的大小 指针大小的计算方式。指针的大小并非由其类型决定&#xff0c;而是与编译器的位数相关。具体来说&#xff0c;在 32 位的系统下&#xff0c;指针的大小为 4 个字节&#xff0c;而在 64 位的系统下&#xff0c;指针的大小为 8 个字节。通过示例代码的运行&#xf…

python可视化:端午假期旅游火爆原因分析

python可视化&#xff1a;端午假期旅游火爆原因分析 2025年的旅游市场表现强劲&#xff1a; 2025年端午假期全社会跨区域人员流动量累计6.57亿人次&#xff0c;日均2.19亿人次&#xff0c;同比增长3.0%。入境游订单同比大涨近90%&#xff0c;门票交易额&#xff08;GMV&#…

ubuntu22.04安装taskfile

sh -c "$(curl --location https://taskfile.dev/install.sh)" -- -dsudo mv ./bin/task /usr/local/bin/测试 task --version

Ubuntu22.04 安装 Miniconda3

Conda 是一个开源的包管理系统和环境管理系统&#xff0c;可用于 Python 环境管理。 Miniconda 是一个轻量级的 Conda 发行版。Miniconda 包含了 Conda、Python和一些基本包&#xff0c;是 Anaconda 的精简版本。 1.下载安装脚本 在 conda官网 找到需要的安装版本&#xff0…

LRC and VIP

//首先排除所有数相等的情况,再把最大值放在一个组&#xff0c;那么最大值的gcd就等于其本身&#xff0c;再判断剩下的gcd是否等于最大值就可以了 #include<bits/stdc.h> using namespace std;const int N1e3100; int a[N]; map<int,int>mapp; int main(){int t;ci…

AI Agent开发第78课-大模型结合Flink构建政务类长公文、长文件、OA应用Agent

开篇 AI Agent2025确定是进入了爆发期,到处都在冒出各种各样的实用AI Agent。很多人、组织都投身于开发AI Agent。 但是从3月份开始业界开始出现了一种这样的声音: AI开发入门并不难,一旦开发完后没法用! 经历过至少一个AI Agent从开发到上线的小伙伴们其实都听到过这种…

统信 UOS 服务器版离线部署 DeepSeek 攻略

日前&#xff0c;DeepSeek 系列模型因拥有“更低的成本、更强的性能、更好的体验”三大核心优势&#xff0c;在全球范围内备受瞩目。 本次&#xff0c;我们为大家提供了在统信 UOS 服务器版 V20&#xff08;AMD64 或 ARM64 架构&#xff09;上本地离线部署 DeepSeek-R1 模型的…

6月2日day43打卡

复习日 作业&#xff1a; kaggle找到一个图像数据集&#xff0c;用cnn网络进行训练并且用grad-cam做可视化 进阶&#xff1a;并拆分成多个文件 任务写了两天&#xff0c;第一天找到一个数据集Stanford Cars Dataset&#xff08;斯坦福汽车数据集&#xff09;&#xff1a; 1. 基…

机器学习——聚类算法

一、聚类的概念 根据样本之间的相似性&#xff0c;将样本划分到不同的类别中的一种无监督学习算法。 细节&#xff1a;根据样本之间的相似性&#xff0c;将样本划分到不同的类别中&#xff1b;不同的相似度计算方法&#xff0c;会得到不同的聚类结果&#xff0c;常用的相似度…

蓝桥杯_DS18B20温度传感器---新手入门级别超级详细解析

目录 一、引言 DS18B20的原理图 单总线简介&#xff1a; ​编辑暂存器简介&#xff1a; DS18B20的温度转换与读取流程 二、代码配置 maic文件 疑问 关于不同格式化输出符号的使用 为什么要rd_temperature()/16.0&#xff1f; onewire.h文件 这个配置为什么要先读lo…

SuperMap GIS基础产品FAQ集锦(20250603)

一、SuperMap iDesktopX 问题1&#xff1a;这种投影坐标如何转换成China_2000的&#xff1f; 11.2.0 【解决办法】在数据源属性中&#xff0c;选择坐标系下的投影转换&#xff0c;然后指定转换结果的坐标系为China_2000 问题2&#xff1a;SuperMap iDesktopX 影像导出时&am…

【js 图片批量自定义打包下载】

压缩图片打包本地下载 一、依赖安转二、函数封装三、打包压缩四、应用五、示例图 一、依赖安转 打包工具 npm install file-saver --save npm install jszip --save二、函数封装 对图片进行处理 function getBase64Image(src) {return new Promise((resolve, reject) > …

如何轻松地将数据从 iPhone传输到iPhone 16

对升级到 iPhone 16 感到兴奋吗&#xff1f;恭喜&#xff01;然而&#xff0c;除了兴奋之外&#xff0c;学习如何将数据从 iPhone 传输到 iPhone 16 也很重要。毕竟&#xff0c;那些重要的联系人、笔记等都是不可或缺的。为了实现轻松的iPhone 到 iPhone 传输&#xff0c;我们总…

Adobe Acrobat——设置PDF打印页面的大小

1. 打开 PDF 文件&#xff1b; 2. 点击菜单栏的 “文件” → “打印”&#xff1b; 3. 在打印对话框中&#xff0c;点击 “属性”&#xff1b; 4. 点击 “布局”→ “高级”&#xff1b; 5. 点击 “纸张规格”&#xff0c;选择 “PostScript 自定义页面大小”&#xff0c;然后…

胜牌™全球成为2026年FIFA世界杯™官方赞助商

胜牌全球将首次与国际足联&#xff08;FIFA&#xff09;旗舰赛事建立合作关系。 此次赞助恰逢美国首个润滑油品牌即将迎来160周年之际&#xff0c;其国际扩张步伐正在加快。 在这项全球顶级赛事筹备期间&#xff0c;胜牌全球将通过各种富有创意的零售和体验活动与球迷互动。 …

mpg123在MSVC编译器中使用。

官网下载&#xff1a; 下载后打开以下窗口程序&#xff1a; 在此窗口程序中打开所下载的mpg123文件夹。在此文件夹中输入以下命令&#xff1a; dumpbin /EXPORTS libsyn123-0.dll > libsyn123-0.exports lib /DEF:libsyn123-0.def /OUT:libsyn123-0.lib /MACHINE:x64其中…

【LangServe部署流程】5 分钟部署你的 AI 服务

目录 一、LangServe简介 二、环境准备 1. 安装必要依赖 2. 编写一个 LangChain 可运行链&#xff08;Runnable&#xff09; 3. 启动 LangServe 服务 4. 启动服务 5. 使用 API 进行调用 三、可选&#xff1a;访问交互式 Swagger 文档 四、基于 LangServe 的 RAG 应用部…

苍穹外卖--HttpClient

1.介绍 HttpClient是Apache Jakarta Common下的子项目&#xff0c;可以用来提供高效的、最新的、功能丰富的支持HTTP协议的客户端编程工具包&#xff0c;并且它支持HTTP协议最新的版本和建议 依赖&#xff1a; 核心API&#xff1a; ①HTTPClient ②HTTPClients ③Closeabl…