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

mybatis-plus使用记录

MyBatis-Plus 学习笔记

一、 快速入门

MyBatis-Plus (MP) 是一个 MyBatis 的增强工具,在 MyBatis 的基础上只做增强不做改变,为简化开发、提高效率而生。

1. 引入 Maven 依赖

要使用 MyBatis-Plus,首先需要在项目的 pom.xml 文件中引入相关依赖。

官网坐标查看:https://baomidou.com/getting-started/install/

代码段

<dependency><groupId>com.baomidou</groupId><artifactId>mybatis-plus-boot-starter</artifactId><version>最新版本</version>
</dependency>

2. Mapper 层开发

与 MyBatis 不同,使用 MP 后,Mapper 接口只需继承 BaseMapper 即可获得强大的 CRUD 功能,无需编写基础的 SQL 语句。

  • 创建 Mapper 接口: 让你的 Mapper 接口继承 BaseMapper<T>
  • 指定泛型: 泛型 T 必须是与数据库表对应的实体类(Entity)。

Java

@Mapper
public interface UserMapper extends BaseMapper<User> {// 复杂的、自定义的SQL可以写在这里
}

3. Service 层开发

在 Service 层,通过 Spring 的依赖注入(如 @Autowired)引入 Mapper,然后就可以直接调用继承自 BaseMapper 的各种方法。

Java

@Service
public class UserServiceImpl implements IUserService {@Autowiredprivate UserMapper userMapper;public User getUserById(Long id) {// 直接调用 BaseMapper 提供的方法return userMapper.selectById(id);}
}

二、 核心概念与配置

1. 实体类与表映射

默认原则:

MP 默认采用驼峰命名与下划线命名的映射规则。 例如,Java 实体类的 userName 字段会自动映射到数据库表的 user_name 字段。

手动映射注解:

当实体类属性名与数据库字段名不一致,或有特殊需求时,可以使用注解进行手动映射。

  • @TableName: 指定实体类对应的表名。
  • @TableId: 声明该字段为主键。
    • type = IdType.AUTO: 设置为主键自增。如果不设置,MP 默认会使用“雪花算法”生成一个长整型的 ID。
  • @TableField: 声明非主键的字段。
    • value = "db_column_name": 指定映射的数据库字段名。
    • exist = false: 表示该字段在数据库表中不存在,只是业务逻辑需要。

何时需要使用TableField:

Java

@Data
@TableName("sys_user") // 指定表名
public class User {@TableId(type = IdType.AUTO) // 主键自增private Long id;@TableField("user_name") // 字段名映射private String name;private Integer age;@TableField(exist = false) // 数据库中无此字段private String gender;
}

2. 常见配置

可以在 application.ymlapplication.properties 中添加 MP 的相关配置。

YAML

mybatis-plus:type-aliases-package: com.example.project.domain # 实体类别名扫描包mapper-locations: classpath*:/mapper/**/*.xml   # Mapper XML 文件位置configuration:map-underscore-to-camel-case: true           # 开启驼峰下划线转换log-impl: org.apache.ibatis.logging.stdout.StdOutImpl # 打印SQL日志

三、 条件构造器 (Wrapper)

条件构造器是 MP 的精髓所在,它能让我们以面向对象的方式构建复杂的 SQL 查询条件,而无需拼接字符串。

1. QueryWrapper

用于构建查询条件(SELECT 语句的 WHERE 部分)。

Java

// 查询年龄大于等于20,且状态为1的用户
QueryWrapper<User> queryWrapper = new QueryWrapper<>();
queryWrapper.ge("age", 20).eq("status", 1).orderByDesc("create_time");List<User> users = userMapper.selectList(queryWrapper);

2. UpdateWrapper

主要用于构建更新条件,可以精细化控制 UPDATE 语句的 SETWHERE 部分。

Java

// 将所有状态为1的用户的年龄更新为18岁
UpdateWrapper<User> updateWrapper = new UpdateWrapper<>();
updateWrapper.set("age", 18)      // 设置要更新的字段和值.eq("status", 1);  // 设置更新条件userMapper.update(null, updateWrapper); // entity传null,表示不使用实体对象更新

3. Lambda 版 Wrapper

LambdaQueryWrapperLambdaUpdateWrapper 是更推荐的使用方式。它们利用 Java 8 的 Lambda 表达式,可以避免硬编码字段名字符串,提高了代码的可读性和安全性。

  • 传统方式: .eq("status", 1)
  • Lambda方式: .eq(User::getStatus, 1)

Java

// 使用 LambdaUpdateWrapper
userMapper.update(null, new LambdaUpdateWrapper<User>().set(User::getAge, 18).eq(User::getStatus, 1)
);

4.自定义sql:

当where条件前面的内容比较难写,需要拼接的时候,用该方式,调用的是自定义的方法

四、 Service 层封装

MP 对 IService 接口也提供了默认实现 ServiceImpl,进一步简化开发。

1. 接口与实现

  • 接口: 业务接口继承 IService<T>
  • 实现类: 继承 ServiceImpl<M, T>,并实现自己的业务接口。

Java

// 接口 IUserService.java
public interface IUserService extends IService<User> {// 自定义复杂业务方法PageResult<UserDTO> getUserListPage(UserQuery query);
}// 实现类 UserServiceImpl.java
@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {// ServiceImpl已经注入了baseMapper,可以直接使用// 实现自定义方法@Overridepublic PageResult<UserDTO> getUserListPage(UserQuery query) {// ... 复杂逻辑long total = count(); // 调用IService提供的方法List<User> users = baseMapper.selectUserWithDetails(query); // 调用Mapper自定义方法// ...}
}

2. 常用方法

ServiceImpl 提供了一系列便捷的单表操作方法:

  • getOne(wrapper): 查询单个对象。
  • list(wrapper): 查询对象列表。
  • count(wrapper): 查询总记录数。
  • save(entity): 新增。
  • updateById(entity): 根据 ID 更新。
  • removeById(id): 根据 ID 删除。

使用示例: 在 Controller 中注入 IUserService 后,可以直接调用这些方法。

五、 分页查询

MP 内置了强大的分页插件,使用非常简单。

1. 添加分页插件配置

需要将分页插件作为一个 Spring Bean 进行配置。

Java

@Configuration
public class MybatisPlusConfig {@Beanpublic MybatisPlusInterceptor mybatisPlusInterceptor() {MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();// 添加分页插件interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL));return interceptor;}
}

2. 在 Service/Controller 中使用

分页查询的核心是构建一个 Page 对象。

Java

public Page<User> findUsersByPage(int pageNum, int pageSize) {// 1. 创建 Page 对象,传入当前页码和每页数量Page<User> page = new Page<>(pageNum, pageSize);// 2. 调用 Mapper 的 selectPage 方法// MP 会自动拦截 SQL,并拼接上 LIMIT 分页语句userMapper.selectPage(page, null); // wrapper为null表示查询所有// 3. page 对象中就包含了分页数据和总数return page;
}

查询结果 Page<T> 对象包含了丰富的分页信息,如:

  • getRecords(): 获取当前页的数据列表。
  • getTotal(): 获取总记录数。
  • getPages(): 获取总页数。
  • getCurrent(): 获取当前页码。
  • getSize(): 获取每页显示数量。

六、使用流程

决策流程:由上至下的选择策略

您可以将其视为一个决策层级,永远从最顶层(最抽象、最方便)的方法开始尝试,只有当前层级无法满足需求时,才往下一個层级移动。

  1. 首先,考虑 **IService** 的内置方法:这个功能是否为一个简单的单表 CRUD?
  2. 如果不是,考虑自定义 **Service** 方法:这个功能是否为一个复杂的业务流程(例如多个步骤、需要事务控制、数据转换)?
  3. **在自定义 **Service** 方法中,考虑 ****BaseMapper**:我是否需要调用 Mapper 层的方法来执行数据库操作?
  4. 最后,考虑自定义 **Mapper** 方法 + SQL:这个查询是否非常复杂(例如多表 JOIN),以至于 Wrapper 无法满足?

1. 何时使用 IService 中封装的方法(默认首选)

这是指 ServiceImpl 中已经为您准备好的方法,如 save(), getById(), removeById(), list(), page(), count() 等。

  • 核心用途:处理所有标准的、单一数据表的“增、删、改、查”(CRUD) 操作。
  • 使用时机
    • 当您执行的只是一个基础的、针对单一数据表的数据操作时。
    • 查询条件可以用 QueryWrapperLambdaQueryWrapper 轻松构建。
    • 这应该是您的第一选择与默认方法,通常由 Controller 层直接调用,或在简单的业务方法中被使用。

示例:在 Controller 中根据 ID 获取用户信息。

  • Java
@GetMapping("/{id}")
public UserDTO findUserById(@PathVariable Long id) {// IService 的 getById 是最完美的选择,因为这是一个简单的单表查询User user = userService.getById(id); //// 可能会搭配 hutool 做对象转换return BeanUtil.copyProperties(user, UserDTO.class); //
}

2. 何时使用自定义 Service 方法(业务逻辑层)

这是在您自己的 IUserService 接口中定义,并在 UserServiceImpl 中实现的方法。

  • 核心用途封装复杂的业务逻辑,让 Controller 层保持简洁。一个业务逻辑通常不仅仅是一次数据库查询。
  • 使用时机
    • 当一个功能需要多个步骤才能完成时(例如:先新增订单,再更新库存)。
    • 当方法需要加入事务控制 (@Transactional) 时。
    • 当逻辑中包含非数据库操作时(例如:调用外部 API、发送邮件、文件处理)。
    • 当需要进行数据转换与组合时(例如:查询多个表,将结果组合成一个视图对象 DTO/VO)。
    • 当您想将一段复杂的查询逻辑重复使用时。

示例:一个复杂的分页查询,它不仅需要查询数据,还可能需要处理查询参数、调用 Mapper 进行复杂查询,并对结果进行封装。

  • Java
// 在 IUserService.java 中定义
public interface IUserService extends IService<User> {PageResult<User> getUserList(QueryParams params);
}// 在 UserServiceImpl.java 中实现
@Service
public class UserServiceImpl extends ServiceImpl<UserMapper, User> implements IUserService {@Overridepublic PageResult<User> getUserList(QueryParams params) {// 1. 处理业务逻辑...// 2. 调用 Mapper 层进行数据库操作List<User> users = baseMapper.selectUserWithDetails(params); //// 3. 调用 IService 的内置方法获取总数long total = count(); //// 4. 封装结果并返回return new PageResult<>(users, total);}
}

3. 何时使用 BaseMapper 中的方法(基础数据访问)

这是 IService 底层实现所依赖的基础 CRUD 方法,例如 selectById(), selectList(), insert() 等。

  • 核心用途:作为 ServiceImpl 与 Mapper 层沟通的桥梁。
  • 使用时机
    • 在现代 MyBatis-Plus 开发中,您很少会直接使用 **baseMapper** 的内置方法(如 baseMapper.selectById()),因为在 ServiceImpl 中可以直接使用 this.getById(),后者更方便且功能一致。
    • baseMapper 对象最主要的用途是:在自定义的 **Service** 方法中,调用您在 **Mapper** 层自定义的方法(即下一点描述的情况)。
  • 示例:如上一个例子所示,baseMapper.selectUserWithDetails(params) 就是通过 baseMapper 来调用在 UserMapper.xml 中自定义的 SQL 查询。

4. 何时使用自定义 Mapper 方法并自定义 SQL(终极武器)

这是在您的 UserMapper 接口中定义一个新方法,并在对应的 XML 文件中编写其完整的 SQL 语句。

  • 核心用途:执行任何 Wrapper 产生不了的复杂 SQL 查询,提供对 SQL 的完全控制权。
  • 使用时机(当 **Wrapper** 不敷使用时)
    • 需要进行多表 **JOIN** 关联查询时。
    • 需要使用复杂的子查询UNION 操作时。
    • 需要使用特定数据库的函数时(例如 MySQL 的 JSON_EXTRACTFIND_IN_SET)。
    • 有非常复杂的 GROUP BYHAVING 条件时。
    • 对于性能要求极高,需要手动优化和调整 SQL 的场景。

示例:在 UserMapper.xml 中定义一个关联查询。

  • XML
<select id="selectUserWithDetails" resultType="com.example.UserVO">SELECTu.id,u.name,d.dept_nameFROMuser uLEFT JOINdepartment d ON u.dept_id = d.idWHEREu.status = #{params.status}
</select>

然后在 ServiceImpl 中通过 baseMapper 调用它。

总结对照表

方法类型核心用途使用时机示例
**IService**
** 内置方法**
标准、单表 CRUD默认首选。所有简单的单表操作。userService.getById(1L);
自定义 **Service**
方法
封装复杂业务逻辑涉及多步骤、事务、外部调用、数据转换等。userService.createUserAndSendEmail(user);
**BaseMapper**
** 内置方法**
IService
的底层实现
较少直接用,主要用于在 Service 中调用自定义 Mapper 方法baseMapper.selectUserWithDetails(params);
自定义 **Mapper**
方法 + SQL
处理 Wrapper
无法应对的复杂 SQL
最后手段。多表 JOIN、复杂子查询、SQL 优化。在 XML 中写 JOIN
查询。

相关文章:

mybatis-plus使用记录

MyBatis-Plus 学习笔记 一、 快速入门 MyBatis-Plus (MP) 是一个 MyBatis 的增强工具&#xff0c;在 MyBatis 的基础上只做增强不做改变&#xff0c;为简化开发、提高效率而生。 1. 引入 Maven 依赖 要使用 MyBatis-Plus&#xff0c;首先需要在项目的 pom.xml 文件中引入相…...

Mcu_Bsdiff_Upgrade

系统架构 概述 MCU BSDiff 升级系统通过使用二进制差分技术&#xff0c;提供了一种在资源受限的微控制器上进行高效固件更新的机制。系统不传输和存储完整的固件映像&#xff0c;而是只处理固件版本之间的差异&#xff0c;从而显著缩小更新包并降低带宽要求。 该架构遵循一个…...

有监督学习——决策树

任务 1、基于iris_data.csv数据&#xff0c;建立决策树模型&#xff0c;评估模型表现; 2、可视化决策树结构; 3、修改min_samples_leaf参数&#xff0c;对比模型结果 代码工具&#xff1a;jupyter notebook 参考资料 20.23 决策树&#xff08;1&#xff09;_哔哩哔哩_bil…...

华为OD机试真题——启动多任务排序(2025B卷:200分)Java/python/JavaScript/C/C++/GO最佳实现

2025 B卷 200分 题型 本专栏内全部题目均提供Java、python、JavaScript、C、C++、GO六种语言的最佳实现方式; 并且每种语言均涵盖详细的问题分析、解题思路、代码实现、代码详解、3个测试用例以及综合分析; 本文收录于专栏:《2025华为OD真题目录+全流程解析+备考攻略+经验分…...

AWS云与第三方通信最佳实践:安全、高效的数据交互方案

引言 在当今的云计算时代,企业经常需要在AWS云环境中存储和处理数据,同时还需要与第三方应用或服务进行数据交互。如何安全、高效地实现这种通信是许多企业面临的挑战。本文将详细探讨几种AWS云与第三方通信的方案,并分析它们的优缺点,帮助您为自己的业务场景选择最佳解决…...

Ubuntu Server 24 设置 WiFi 网络的方案

一、配置流程 1. 确认无线网卡信息 首先需明确无线网卡接口名称及当前连接状态&#xff1a; ip link show # 查看网络接口&#xff08;寻找状态为 "UP" 的无线接口&#xff0c;如 wlan0、wlx* 或 wlp1s0&#xff09; iwconfig # 确认无线网…...

【redis】redis和hiredis的基本使用

总结&#xff1a; 介绍了一下redis和hiredis的安装步骤&#xff0c;用一个简单的demo演示了使用redis的基本过程。 启动redis步骤 1、下载redis&#xff1a;https://github.com/redis/redis 2、编译命令&#xff1a;make 3、编译产物&#xff1a;libredis.a&#xff08;静…...

大模型时代,Python 近红外光谱与 Transformer 模型:学习的必要性探究

在当下大语言模型盛行的时代&#xff0c;各类新技术如潮水般不断涌现&#xff0c;让人应接不暇。身处这样的浪潮之中&#xff0c;不少人心中都会泛起疑问&#xff1a;Python 近红外光谱和 Transformer 模型还有学习的必要性吗&#xff1f;今天&#xff0c;就让我们深入探讨一番…...

产品经理常用术语大全

作为一名产品经理&#xff0c;不仅需要具备跨领域的知识和技能&#xff0c;还需要熟练掌握一系列专业术语&#xff0c;以便更有效地沟通、规划和执行产品开发过程中的各项任务。以下是一篇详细介绍产品经理日常工作中常见术语的文章&#xff0c;旨在帮助新手快速入门&#xff0…...

梯度优化提示词:精准引导AI分类

基于梯度优化的提示词工程方法,通过迭代调整提示词的嵌入向量,使其能够更有效地引导模型做出正确分类。 数据形式 训练数据 train_data 是一个列表,每个元素是一个字典,包含两个键: text: 需要分类的文本描述label: 对应的标签(“冲动"或"理性”)示例数据: …...

AUTOSAR 运行时环境 (RTE)

目录 往期推荐 什么是运行时环境&#xff1f; AUTOSAR 中的运行时环境 (RTE) RTE 的应用 RTE 的生成 关于RTE API的一些信息 RTE生成后文件之间的关系 往期推荐 2025汽车行业新宠&#xff1a;欧企都在用的工具软件ETAS工具链自动化实战指南&#xff1c;一&#xff1e;ET…...

Bolt.new:重塑 Web 开发格局的 AI 利器

根据 Menlo Ventures 2024 年的调查&#xff0c;在主流 AI 应用场景中&#xff0c;AI 编程工具的采用率以 51% 位居榜首&#xff0c;代码生成成为最易落地且受欢迎的场景。科技巨头谷歌 CEO Sundar Pichai 在 2024 年 10 月财报会议上透露&#xff0c;公司超四分之一的新代码由…...

RK3588 RKNN ResNet50推理测试

RK3588 RKNN ResNet50推理测试 一、背景二、性能数据三、操作步骤3.1 安装依赖3.2 安装rknn-toolkit,更新librknnrt.so3.3 下载推理图片3.4 生成`onnx`模型转换脚本3.5 生成rknn模型3.6 运行rknn模型一、背景 在嵌入式设备上进行AI推理时,我们面临着算力有限、功耗敏感等挑战…...

SQLMesh 宏操作符详解:提升 SQL 查询的灵活性与效率

SQLMesh 提供了一系列强大的宏操作符&#xff08;如 WITH、JOIN、WHERE 等&#xff09;&#xff0c;用于动态构建 SQL 查询。这些操作符不仅简化了复杂查询的编写&#xff0c;还提高了代码的可读性和可维护性。本文将深入探讨这些操作符的使用场景、语法及实际案例&#xff0c;…...

leetcode513.找树左下角的值:递归深度优先搜索中的最左节点追踪之道

一、题目本质与核心诉求解析 在二叉树算法问题中&#xff0c;"找树左下角的值"是一个典型的结合深度与位置判断的问题。题目要求我们找到二叉树中最深层最左边的节点值&#xff0c;这里的"左下角"有两个关键限定&#xff1a; 深度优先&#xff1a;必须是…...

基于Flink的数据中台管理平台

基于Flink做的数据中台工程项目。数据从source到clickhouse全流程的验证。集成元数据管、数据资产、数据发现功能&#xff0c;自主管理元数据变更&#xff0c;集成元数据版本管理。 同时&#xff0c;对整个大数据集群使用到的组件或者是工具进行管理。比如nacos、kafka、zookee…...

AI-Ready TapData:如何基于 MCP 协构建企业级 AI 实时数据中枢?(含教程)

随着企业对私有大模型、行业大模型的探索逐渐深入&#xff0c;“AI应用是否真正落地”&#xff0c;越来越取决于企业是否拥有结构化、实时、可交互的高质量数据。而现实是&#xff0c;大多数企业的核心业务数据依旧被困在多个异构系统、孤岛数据库和 ETL 流程之中&#xff0c;导…...

Spring Boot 登录实现:JWT 与 Session 全面对比与实战讲解

Spring Boot 登录实现&#xff1a;JWT 与 Session 全面对比与实战讲解 2025.5.21-23:11今天在学习黑马点评时突然发现用的是与苍穹外卖jwt不一样的登录方式-Session&#xff0c;于是就想记录一下这两种方式有什么不同 在实际开发中&#xff0c;登录认证是后端最基础也是最重要…...

【HTML-5】HTML 实体:完整指南与最佳实践

1. 什么是 HTML 实体&#xff1f; HTML 实体是一种在 HTML 文档中表示特殊字符的方法&#xff0c;这些字符如果直接使用可能会与 HTML 标记混淆&#xff0c;或者无法通过键盘直接输入。实体由 & 符号开始&#xff0c;以 ; 分号结束。 <p>这是一个小于符号的实体&am…...

SpringBoot 项目实现操作日志的记录(使用 AOP 注解模式)

本文是博主在做关于如何记录用户操作日志时做的记录&#xff0c;常见的项目中难免存在一些需要记录重要日志的部分&#xff0c;例如权限和角色设定&#xff0c;重要数据的操作等部分。 博主使用 Spring 中的 AOP 功能&#xff0c;结合注解的方式&#xff0c;对用户操作过的一些…...

AI|Java开发 IntelliJ IDEA中接入本地部署的deepseek方法

目录 连接本地部署的deepseek&#xff1a; IntelliJ IDEA中使用deepseek等AI&#xff1a; 用法一&#xff1a;让AI写代码 用法二&#xff1a;选中这段代码&#xff0c;右键&#xff0c;可以让其解释这段代码的含义。这时显示的解释是英文的。 连接本地部署的deepseek&#…...

【疑难杂症】Vue前端下载文件无法打开 已解决

由于刚学了VUE不久&#xff0c;不清楚底层逻辑。我遇到从后台下载文件无法打开的问题。 测试下来是&#xff0c;请求时未设置 responseType: blob。 axios 默认的 responseType 是 json &#xff0c;会尝试将响应体解析为JSON。但文件下载场景需要后端返回二进制流&#xff0…...

【1——Android端添加隐私协议(unity)1/3】

前言&#xff1a;这篇仅对于unity 发布Android端上架国内应用商店添加隐私协议&#xff0c;隐私协议是很重要的东西&#xff0c;没有这个东西&#xff0c;是不上了应用商店的。 对于仅仅添加隐私协议&#xff0c;我知道有三种方式,第一种和第二种基本一样 1.直接在unity里面新…...

Linux之概述和安装vm虚拟机

文章目录 操作系统概述硬件和软件操作系统常见操作系统 初识LinuxLinux的诞生Linux内核Linux发行版 虚拟机介绍虚拟机 VMware WorkStation安装虚拟化软件VMware WorkStation 安装查看VM网络连接设置VM存储位置 在VMware上安装Linux(发行版CentOS7)安装包获取CentOS7 安装 Mac系…...

深入理解 Linux 的 set、env 和 printenv 命令

在 Linux 和类 Unix 系统中&#xff0c;环境变量是配置和管理 Shell 及进程行为的核心机制。set、env 和 printenv 是与环境变量交互的三个重要命令&#xff0c;每个命令都有其独特的功能和用途。本文将详细探讨这三个命令的区别&#xff0c;帮助大家更好地理解和使用这些命令。…...

LeetCode热题100--19.删除链表的倒数第N个结点--中等

1. 题目 给你一个链表&#xff0c;删除链表的倒数第 n 个结点&#xff0c;并且返回链表的头结点。 示例 1&#xff1a; 输入&#xff1a;head [1,2,3,4,5], n 2 输出&#xff1a;[1,2,3,5] 示例 2&#xff1a; 输入&#xff1a;head [1], n 1 输出&#xff1a;[] 示例…...

开发AR导航助手:ARKit+Unity+Mapbox全流程实战教程

引言 在增强现实技术飞速发展的今天&#xff0c;AR导航应用正逐步改变人们的出行方式。本文将手把手教你使用UnityARKitMapbox开发跨平台AR导航助手&#xff0c;实现从虚拟路径叠加到空间感知的完整技术闭环。通过本教程&#xff0c;你将掌握&#xff1a; AR空间映射与场景理…...

git学习与使用(远程仓库、分支、工作流)

文章目录 前言简介git的工作流程git的安装配置git环境&#xff1a;git config --globalgit的基本使用新建目录初始化仓库&#xff08;repository&#xff09;添加到暂存区新增/修改/删除 文件状态会改变 提交到仓库查看提交&#xff08;commit&#xff09;的历史记录git其他命令…...

嵌入式预处理链接脚本lds和map文件

在嵌入式开发中&#xff0c;.lds.S 文件是一个 预处理后的链接脚本&#xff08;Linker Script&#xff09;&#xff0c;它结合了 C 预处理器&#xff08;Preprocessor&#xff09; 的功能和链接脚本的语法。它的核心作用仍然是 定义内存布局和链接规则&#xff0c;但通过预处理…...

9. Spring AI 各版本的详细功能与发布时间整理

目录 一、旧版本(Legacy) 0.8.1(2024年3月) 二、里程碑版本(Milestone) 1.0.0-M1(2024年5月30日) 1.0.0-M2(2024年7月) 1.0.0-M3(2024年10月8日) 1.0.0-M4(2024年12月) 1.0.0-M5(2025年1月9日) 1.0.0-M6(2025年3月) 1.0.0-M7(2025年4月14日) 1.…...