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

SpringBoot 声明式事务与编程式事务

上一篇我们详细讲解了 Transactional 注解的用法、原理和失效场景其实 Transactional 属于「声明式事务」是 SpringBoot 中最常用的事务管理方式。但很多开发者不知道SpringBoot 还支持另一种事务管理方式——「编程式事务」。实际开发中经常有同学纠结什么时候用声明式事务什么时候用编程式事务两者到底有什么区别选错了不仅会增加开发成本还可能导致事务失效、数据不一致等问题。一、什么是声明式事务什么是编程式事务在对比之前我们先明确两种事务的核心定义用通俗的语言拆解不用死记硬背重点理解「使用方式」的差异。1. 声明式事务声明式事务是 SpringBoot 中最主流、最常用的事务管理方式核心是「通过注解或 XML 配置声明事务」无需手动编写事务开启、提交、回滚的代码由 Spring 自动管理事务生命周期。简单来说只需要在方法或类上添加 Transactional 注解Spring 就会自动为该方法添加事务支持开发者只需关注核心业务逻辑无需关心事务的底层实现。核心特点无侵入式、配置简单、代码简洁符合 Spring「约定大于配置」的理念适合 90% 以上的企业级开发场景。✅ 代码示例Service public class OrderService { // 声明式事务一个注解搞定事务管理 Transactional(rollbackFor Exception.class) public void createOrder(OrderDTO orderDTO) { // 核心业务逻辑扣库存、生成订单、扣余额 stockMapper.decreaseStock(orderDTO.getProductId(), orderDTO.getQuantity()); orderMapper.insert(buildOrder(orderDTO)); userMapper.decreaseBalance(orderDTO.getUserId(), orderDTO.getTotalAmount()); } }2. 编程式事务编程式事务是「通过手动编写代码控制事务」开发者需要手动调用 Spring 提供的 API完成事务的开启、提交、回滚操作事务的每一个环节都需要手动控制。简单来说事务的生命周期完全由开发者掌控需要手动写代码开启事务、执行业务逻辑、判断是否提交或回滚侵入业务代码但灵活性极高。核心特点侵入式、配置复杂、代码繁琐但灵活性强适合复杂的事务场景比如多事务嵌套、动态控制事务开关。✅ 代码示例使用 TransactionTemplateService public class OrderService { Autowired private TransactionTemplate transactionTemplate; // 编程式事务手动控制事务的开启、提交、回滚 public void createOrder(OrderDTO orderDTO) { // 手动开启事务执行核心逻辑 transactionTemplate.execute(status - { try { // 核心业务逻辑 stockMapper.decreaseStock(orderDTO.getProductId(), orderDTO.getQuantity()); orderMapper.insert(buildOrder(orderDTO)); userMapper.decreaseBalance(orderDTO.getUserId(), orderDTO.getTotalAmount()); return true; // 执行成功自动提交事务 } catch (Exception e) { status.setRollbackOnly(); // 执行失败手动回滚事务 throw new RuntimeException(e.getMessage()); } }); } }补充说明编程式事务有两种实现方式① 使用 TransactionTemplate推荐简化代码② 使用 PlatformTransactionManager原生 API代码繁琐几乎不用。本篇重点讲解 TransactionTemplate 的用法。二、底层原理对比两种事务管理方式的底层都依赖 Spring 的「事务管理器PlatformTransactionManager」但实现逻辑、触发方式有本质区别这也是两者优缺点、适用场景差异的核心原因。1. 声明式事务原理声明式事务的底层是Spring AOP 动态代理和 Transactional 注解的原理一致核心流程如下1. Spring 容器启动时扫描带有 Transactional 注解的类或方法为其生成动态代理对象JDK 动态代理或 CGLIB 动态代理2. 调用目标方法时实际上调用的是代理对象的方法3. 代理对象在方法执行前通过事务管理器开启事务设置事务属性传播行为、隔离级别等4. 执行目标方法的核心业务逻辑5. 方法正常执行代理对象通过事务管理器提交事务6. 方法抛出异常代理对象通过事务管理器回滚事务根据 rollbackFor 配置判断是否回滚。核心结论声明式事务是「AOP 增强 事务管理器」的结合无需手动干预Spring 自动完成事务生命周期管理。2. 编程式事务原理编程式事务的底层是手动调用事务管理器 API开发者直接通过代码控制事务的每一个环节核心流程如下1. 开发者通过 TransactionTemplate 或 PlatformTransactionManager手动触发事务开启2. 在事务范围内执行核心业务逻辑3. 手动判断业务逻辑执行结果成功则提交事务失败则手动触发回滚4. 事务执行完成后手动释放资源Spring 会自动处理但开发者可手动控制。核心结论编程式事务是「手动调用 API 事务管理器」的结合事务的每一个环节都由开发者掌控不依赖 AOP 动态代理因此不会出现 AOP 代理相关的失效问题。三、全方位对比这是本篇文章的核心内容我们从「使用方式、代码侵入性、灵活性、配置复杂度、事务控制粒度、失效风险、性能、适用场景」8个核心维度对两种事务做全面对比用表格呈现清晰易懂方便选型和记忆。对比维度声明式事务Transactional编程式事务TransactionTemplate使用方式通过 Transactional 注解或 XML 配置声明无需手动写事务代码通过 TransactionTemplate 手动编写代码控制事务开启、提交、回滚代码侵入性无侵入式仅添加注解不修改业务代码侵入式事务代码与业务代码耦合在一起灵活性较低事务属性传播行为、隔离级别等固定配置无法动态调整极高可动态控制事务开关、提交/回滚时机支持复杂的事务逻辑如多事务嵌套配置复杂度极低单数据源无需额外配置多数据源只需指定事务管理器较高需要注入 TransactionTemplate手动编写事务执行逻辑代码繁琐事务控制粒度较粗只能控制类或方法级别无法控制方法内的某一段代码极细可控制方法内的任意一段代码比如某几行代码需要事务其他代码不需要失效风险较高容易出现 AOP 代理相关的失效问题如 private 方法、同类内部调用、未配置 rollbackFor 等极低不依赖 AOP 代理手动控制事务只要代码编写正确几乎不会失效性能稍低依赖 AOP 动态代理有轻微的性能损耗可忽略不影响业务稍高无 AOP 代理损耗手动控制事务性能略优于声明式事务适用场景大部分常规场景单事务、简单业务逻辑如下单、支付、新增数据追求代码简洁、开发高效复杂事务场景多事务嵌套、动态控制事务、方法内局部代码需要事务、对事务粒度要求极高的场景重点提醒不要盲目追求编程式事务的「高性能」和「高灵活性」大部分场景下声明式事务的性能损耗可以忽略且开发效率更高、代码更简洁只有在声明式事务无法满足需求时才考虑使用编程式事务。四、两种事务的完整用法结合「下单场景」分别用声明式事务和编程式事务实现对比两种方式的代码差异方便大家在项目中直接复用。1. 环境准备单数据源导入依赖SpringBoot 2.x配置 application.yml 数据源此处省略和上一篇事务实战的环境完全一致可直接复用。2. 声明式事务实战需求下单操作扣库存、生成订单、扣余额三个操作必须在同一个事务中任意一个失败全部回滚日志记录用独立事务即使主事务回滚日志也要保存。Service Slf4j public class OrderService { Autowired private OrderMapper orderMapper; Autowired private StockMapper stockMapper; Autowired private UserMapper userMapper; Autowired private OrderLogService orderLogService; // 声明式事务主事务默认传播行为 REQUIRED Transactional(rollbackFor Exception.class, timeout 5) public void createOrder(OrderDTO orderDTO) { try { // 1. 扣减库存 int stockRows stockMapper.decreaseStock(orderDTO.getProductId(), orderDTO.getQuantity()); if (stockRows 0) { throw new BusinessException(库存不足); } // 2. 生成订单 Order order buildOrder(orderDTO); orderMapper.insert(order); // 3. 扣减余额 int userRows userMapper.decreaseBalance(orderDTO.getUserId(), orderDTO.getTotalAmount()); if (userRows 0) { throw new BusinessException(用户余额不足); } // 4. 记录日志独立事务 orderLogService.recordOrderLog(order.getOrderNo(), 下单成功); } catch (Exception e) { log.error(下单失败{}, e.getMessage(), e); throw new RuntimeException(e.getMessage()); } } // 构建订单对象辅助方法 private Order buildOrder(OrderDTO orderDTO) { Order order new Order(); order.setUserId(orderDTO.getUserId()); order.setProductId(orderDTO.getProductId()); order.setQuantity(orderDTO.getQuantity()); order.setOrderNo(UUID.randomUUID().toString().replace(-, )); order.setCreateTime(new Date()); return order; } } // 日志 Service独立事务声明式 Service public class OrderLogService { Autowired private OrderLogMapper orderLogMapper; Transactional(rollbackFor Exception.class, propagation Propagation.REQUIRES_NEW) public void recordOrderLog(String orderNo, String content) { OrderLog log new OrderLog(); log.setOrderNo(orderNo); log.setContent(content); log.setCreateTime(new Date()); orderLogMapper.insert(log); } }✅ 特点代码简洁仅通过注解控制事务无需关注事务的底层实现开发效率高。3. 编程式事务需求下单操作分两步① 扣库存、生成订单必须在事务中② 扣减余额可选事务根据用户配置动态决定是否开启事务。这个场景用声明式事务无法实现无法动态控制事务开关必须用编程式事务Service Slf4j public class OrderService { Autowired private OrderMapper orderMapper; Autowired private StockMapper stockMapper; Autowired private UserMapper userMapper; Autowired private TransactionTemplate transactionTemplate; // 编程式事务动态控制事务开关 public void createOrder(OrderDTO orderDTO, boolean needBalanceTransaction) { // 第一步扣库存、生成订单必须在事务中 Order order transactionTemplate.execute(status - { try { // 扣减库存 int stockRows stockMapper.decreaseStock(orderDTO.getProductId(), orderDTO.getQuantity()); if (stockRows 0) { status.setRollbackOnly(); // 手动回滚 throw new BusinessException(库存不足); } // 生成订单 Order newOrder buildOrder(orderDTO); orderMapper.insert(newOrder); return newOrder; // 事务提交返回订单对象 } catch (Exception e) { status.setRollbackOnly(); log.error(扣库存、生成订单失败{}, e.getMessage()); throw new RuntimeException(e.getMessage()); } }); // 第二步扣减余额动态控制是否开启事务 if (needBalanceTransaction) { // 开启事务扣减余额 transactionTemplate.execute(status - { try { int userRows userMapper.decreaseBalance(orderDTO.getUserId(), orderDTO.getTotalAmount()); if (userRows 0) { status.setRollbackOnly(); throw new BusinessException(用户余额不足); } return true; } catch (Exception e) { status.setRollbackOnly(); log.error(扣减余额失败{}, e.getMessage()); throw new RuntimeException(e.getMessage()); } }); } else { // 不开启事务直接扣减余额非事务操作 userMapper.decreaseBalance(orderDTO.getUserId(), orderDTO.getTotalAmount()); } } private Order buildOrder(OrderDTO orderDTO) { // 同声明式事务的辅助方法省略 Order order new Order(); order.setUserId(orderDTO.getUserId()); order.setProductId(orderDTO.getProductId()); order.setQuantity(orderDTO.getQuantity()); order.setOrderNo(UUID.randomUUID().toString().replace(-, )); order.setCreateTime(new Date()); return order; } }✅ 特点灵活性极高可根据参数needBalanceTransaction动态控制是否开启事务还能分步骤控制事务范围这是声明式事务无法实现的。五、注意事项1. 声明式事务上一篇我们详细讲解了声明式事务的8种失效场景这里重点回顾3个最常见的• 不要在 private、protected 方法上使用 Transactional无法被 AOP 代理事务失效• 同类内部调用时不要用 this.方法()调用的是目标对象不是代理对象事务失效• 务必配置 rollbackFor Exception.class避免 checked 异常不回滚。2. 编程式事务• 不要忘记手动设置 rollbackOnly业务逻辑失败时必须调用 status.setRollbackOnly()否则事务会自动提交• 避免事务嵌套时的资源泄露Spring 的 TransactionTemplate 会自动释放资源但手动使用 PlatformTransactionManager 时需手动释放• 不要过度使用编程式事务简单场景用编程式事务会增加代码复杂度降低开发效率。八、文末小结声明式事务和编程式事务没有绝对的优劣之分核心是「适配业务场景」1. 声明式事务简单、高效、无侵入是日常开发的首选适合大部分常规业务2. 编程式事务灵活、可控、低失效风险适合复杂事务场景是声明式事务的补充。核心选型原则「能⽤声明式就不用编程式」只有在声明式事务无法满足需求时再考虑编程式事务。同时无论使用哪种事务都要牢记避坑点确保事务生效保证数据一致性。如果你在项目中遇到事务选型或事务失效的问题或者有其他疑问欢迎在评论区留言交流一起避坑、一起进步别忘了点赞在看收藏三连关注我解锁更多 SpringBoot 实战干货下期再见❤️

相关文章:

SpringBoot 声明式事务与编程式事务

上一篇我们详细讲解了 Transactional 注解的用法、原理和失效场景,其实 Transactional 属于「声明式事务」,是 SpringBoot 中最常用的事务管理方式。但很多开发者不知道,SpringBoot 还支持另一种事务管理方式——「编程式事务」。实际开发中&…...

3步搞定AI视频生成:ComfyUI-WanVideoWrapper终极入门指南

3步搞定AI视频生成:ComfyUI-WanVideoWrapper终极入门指南 【免费下载链接】ComfyUI-WanVideoWrapper 项目地址: https://gitcode.com/GitHub_Trending/co/ComfyUI-WanVideoWrapper 想要在ComfyUI中轻松实现AI视频生成?ComfyUI-WanVideoWrapper是…...

暗黑破坏神2存档编辑终极指南:使用d2s-editor打造完美角色

暗黑破坏神2存档编辑终极指南:使用d2s-editor打造完美角色 【免费下载链接】d2s-editor 项目地址: https://gitcode.com/gh_mirrors/d2/d2s-editor 你是否想在暗黑破坏神2中自由调整角色属性、打造理想装备、优化游戏进度?d2s-editor为你提供了完…...

区块链应用·数据共享消除数字鸿沟

基于FISCO BCOS与Go语言构建可信数据共享基础设施,打通跨机构、跨地域的信任壁垒 一、数字鸿沟的根源:信任缺失下的“数据孤岛” 数字鸿沟(Digital Divide)不仅存在于不同区域、不同群体之间,更深层次地体现在数据持有者之间的信任鸿沟。在传统信息系统中,数据分散存储于…...

告别玄学调参!基于STM32G4的PID与PFC算法调试实录:我是如何用示波器和串口把效率做到95%+的

STM32G4实战:从波形捕获到参数优化,我的95%效率电源调参手记 实验室的示波器屏幕上,PWM波形正在不规则地抖动,电源模块发出轻微的啸叫声——这熟悉的一幕让我意识到,又一次PID参数调试马拉松开始了。作为嵌入式工程师&…...

基于STM32的平衡机器人PID控制系统设计

一、系统概述与核心原理 1. 系统定位 基于STM32的两轮自平衡机器人(Balance Bot)是自动控制理论的经典实践平台。系统通过MPU6050陀螺仪实时监测车身倾角,利用PID算法计算出电机补偿量,驱动直流电机保持车身直立不倒,并…...

我为什么鼓励团队成员写技术博客?

我为什么鼓励团队成员写技术博客? 在技术团队中,知识沉淀与分享是推动个人和团队成长的重要方式。作为一名技术管理者,我始终鼓励团队成员撰写技术博客,这不仅是为了提升个人影响力,更是为了构建团队的技术文化。那么…...

成都办公室租赁,揭秘行业领先者

在成都这座充满活力的城市,越来越多的企业选择在这里扎根和发展。然而,在企业选址、办公空间租赁等方面,却面临着诸多挑战。信息不对称、资源分散、谈判不专业等问题常常困扰着企业主们。那么,在众多的办公租赁服务商中&#xff0…...

GitHub中文界面终极指南:3分钟让英文GitHub变身中文工作台

GitHub中文界面终极指南:3分钟让英文GitHub变身中文工作台 【免费下载链接】github-chinese GitHub 汉化插件,GitHub 中文化界面。 (GitHub Translation To Chinese) 项目地址: https://gitcode.com/gh_mirrors/gi/github-chinese 你是否曾因GitH…...

Halcon 实战指南:基于局部形变的模板匹配在柔性物体检测中的应用与参数调优

1. 柔性物体检测的挑战与局部形变匹配的价值 在工业视觉检测中,软包装、纺织品、橡胶件等柔性物体的检测一直是个难题。这些材料在传送带或机械臂抓取过程中,难免会发生拉伸、褶皱等轻微形变。传统的刚性模板匹配方法在这里往往会失效——因为哪怕1%的形…...

【2026年最新600套毕设项目分享】基于微信小程序的社区团购(30096)

有需要的同学,源代码和配套文档领取,加文章最下方的名片哦 一、项目演示 项目演示视频 二、资料介绍 完整源代码(前后端源代码SQL脚本)配套文档(LWPPT开题报告/任务书)远程调试控屏包运行一键启动项目&…...

MaaYuan:重塑手游日常体验的智能自动化革命

MaaYuan:重塑手游日常体验的智能自动化革命 【免费下载链接】MaaYuan 代号鸢 / 如鸢 一键长草小助手 项目地址: https://gitcode.com/gh_mirrors/ma/MaaYuan 在现代手游生态中,玩家常常陷入一个矛盾境地:游戏内容日益丰富,…...

黎阳之光核工厂202应急管控平台|全域实景孪生,筑牢核安全最后一道防线

核安全是国家安全的重要组成部分,核工厂应急管控直接关系人员安全、环境安全与社会稳定。面对事故预警难、定位不准、视频割裂、数据孤岛、指挥滞后、追溯困难等行业痛点,北京黎阳之光依托Sinotoon全域实景引擎,重磅推出核工厂202应急管控平台…...

树莓派4B网络启动后,如何用NFS挂载实现多台Pi共享一个系统镜像?

树莓派4B网络启动进阶:NFS共享系统镜像的多设备部署方案 当实验室里摆放着二十台树莓派,每台都需要相同的开发环境时,传统SD卡烧录方式立刻暴露出效率短板。想象一下:系统升级需要逐台插拔卡片,配置变更要重复操作二十…...

vue2+element-UI表格封装

针对表格进行封装&#xff0c;在列表页面直接传入字段数组就可以展示数据表&#xff1a; <template><div class"table-container" :class"{ show-vertical-lines: showVerticalLines }"><!-- 数据表格 --><el-table ref"tableRe…...

MIPI-DSI协议解析:从物理层到应用层的LCD驱动实践

1. MIPI-DSI协议基础&#xff1a;显示接口的"高速公路" 第一次接触MIPI-DSI时&#xff0c;我盯着开发板上那几根细如发丝的走线发愣——这么少的线路怎么传输高清图像&#xff1f;后来才明白&#xff0c;这正是MIPI-DSI的精妙之处。作为移动设备显示接口的事实标准&…...

从PID调参到根轨迹:一个电机控制工程师的实战避坑笔记

从PID调参到根轨迹&#xff1a;一个电机控制工程师的实战避坑笔记 作为一名在工业自动化领域摸爬滚打多年的电机控制工程师&#xff0c;我深知PID参数调试的痛点和挑战。每当面对一个全新的电机控制系统&#xff0c;传统的试凑法不仅耗时耗力&#xff0c;还常常陷入"调好一…...

MT7916芯片深度解析:从拆机中兴E1630看MTK首款AX3000方案

1. 拆机中兴E1630&#xff1a;MT7916芯片的首次亮相 最近在闲鱼上看到中兴E1630这款路由器&#xff0c;价格209元入手&#xff0c;虽然有点小贵&#xff0c;但为了第一时间给大家带来拆机评测还是值得的。这款中国电信定制版路由器外包装略显陈旧&#xff0c;但内部设备保存完好…...

软件测试核心概念实战解析:从理论到习题的深度贯通

1. 软件测试基础理论的核心要点 软件测试作为软件开发过程中不可或缺的一环&#xff0c;其理论基础直接影响着测试工作的质量和效率。在软件测试领域&#xff0c;有几个核心概念是每个测试人员都必须掌握的。 首先&#xff0c;我们需要理解软件生命周期这个概念。简单来说&…...

OP-TEE安全存储深度解析(一):密钥层级与文件加密流程

1. 密钥管理器的核心角色 在OP-TEE的安全存储架构中&#xff0c;密钥管理器就像是一个高度戒备的金库管理员。它不直接存储用户数据&#xff0c;而是负责生成、保护和调度所有用于加密的密钥。这个设计非常巧妙——即使攻击者突破了外层防御&#xff0c;拿到的也只是加密后的数…...

ESP32-C3 蓝牙应用实战:从零构建一个自定义 GATT 服务

1. 环境准备&#xff1a;搭建ESP32-C3开发环境 第一次接触ESP32-C3蓝牙开发的朋友可能会觉得无从下手&#xff0c;其实只要把开发环境搭好就成功了一半。我刚开始用Windows系统开发时踩过不少坑&#xff0c;后来发现用VSCode配合官方ESP-IDF工具链最省心。 首先需要安装乐鑫官方…...

6.6 实战解析——破解可转债数据爬取难题(XPath精准定位与Selenium登录失效的应对策略)

1. 可转债数据爬取的核心挑战 最近在做一个金融数据分析项目时&#xff0c;需要获取可转债的实时行情数据。我首先想到的就是从集思录这类专业网站抓取数据&#xff0c;但实际操作中发现几个棘手的问题。最让人头疼的是&#xff0c;明明用Selenium模拟登录成功了&#xff0c;却…...

ISO 15118-20:2022 深度解读:第二代车网通信接口如何重塑智能充电与电网互动

1. ISO 15118-20:2022标准的前世今生 第一次听说ISO 15118这个标准时&#xff0c;我正蹲在充电站调试一台死活连不上充电桩的电动车。当时满脑子都是"为什么连个充电都要搞这么复杂&#xff1f;"后来才知道&#xff0c;这背后藏着整个电动汽车与电网对话的密码。ISO…...

Unity Shader 中 ShadowCaster的作用和疑问

1. ShadowCaster&#xff08;核心 —— 让物体能投影&#xff09;LightMode ShadowCaster&#xff1a;URP 渲染 Shadow Map 时会调用此 PassColorMask 0&#xff1a;不输出颜色&#xff0c;只写深度顶点关键步骤&#xff1a;TransformObjectToWorld / TransformObjectToWorldN…...

计算机毕业设计:Python农作物产量智能预估与数据看板 Flask框架 XGBoost 机器学习 数据分析 可视化 大数据 大模型(建议收藏)✅

1、项目介绍 技术栈 采用 Python 语言开发&#xff0c;基于 Flask 框架搭建后端服务&#xff0c;使用 MySQL 数据库进行数据存储&#xff0c;通过 pymysql 连接数据库&#xff0c;运用 XGBoost 机器学习模型实现产量预测&#xff0c;前端结合 HTML、CSS、JavaScript、Echarts 和…...

【智能代码生成×代码度量双引擎实战指南】:20年架构师亲授如何用AI写代码+量化质量,规避97%的交付返工风险

第一章&#xff1a;智能代码生成代码度量双引擎协同范式 2026奇点智能技术大会(https://ml-summit.org) 传统AI编程辅助工具常将代码生成与质量评估割裂为独立流程&#xff1a;生成模型输出后&#xff0c;再由静态分析器进行滞后性度量。双引擎协同范式则打破这一时序壁垒&…...

从源码到实战:手把手教你编译与定制化iperf网络性能测试工具

1. iperf工具简介与适用场景 iperf是一款经典的开源网络性能测试工具&#xff0c;它通过测量TCP/UDP带宽来评估网络质量。我第一次接触这个工具是在调试嵌入式设备的网络吞吐量时&#xff0c;当时需要验证百兆网口的实际传输速率是否达标。相比简单的ping命令&#xff0c;iperf…...

基于西门子PLCS7-1200的程序仿真立体车库设计报告(含硬件原理图和CAD)

立体车库设计&#xff0c;基于西门子plcs7-1200带程序仿真&#xff0c;报告&#xff08;过1w&#xff09;&#xff0c;硬件原理图和cad 功能具体如下: 地面层配备七个停车位的升降系统能够有效执行车位的垂直转移在该层&#xff0c;四个停车位安装有自动升降装置&#xff0…...

【雷达成像】基于二维ADMM的稀度驱动ISAR成像附Matlab复现含文献

​✅作者简介&#xff1a;热爱科研的Matlab仿真开发者&#xff0c;擅长毕业设计辅导、数学建模、数据处理、建模仿真、程序设计、完整代码获取、论文复现及科研仿真。&#x1f34e; 往期回顾关注个人主页&#xff1a;Matlab科研工作室&#x1f447; 关注我领取海量matlab电子书…...

hermes agent 初体验

一、安装 参考链接&#xff1a;https://dashscope.aliyuncs.com/compatible-mode 二、报错及解决 问题1&#xff1a;WSL2安装报错 我是WSL2安装的&#xff0c;期间报错&#xff1a;Failed to install uv。我并没有按照他的要求手动去安装&#xff0c;而是再次执行&#xff1a;c…...