大二学完 MyBatis 再学 MyBatis-Plus,我踩过的 10 个坑

news2026/5/22 6:35:29
作者逆境不可逃技术永无止境希望我的内容可以帮助到你本节目属于专栏《后端新手谈》https://blog.csdn.net/2401_87662859/category_13141790.html大家吼 ! 我是 逆境不可逃 今天给大家带来文章《后端新手谈第十五期》《大二学完 MyBatis 再学 MyBatis-Plus我踩过的 10 个坑》.前言大二上学期我在课程设计里第一次用了 MyBatis。当时的感觉是写个用户管理模块增删改查四个功能Mapper 接口加 XML 写了一百多行。更崩溃的是后来老师说要加一个字段我从 Entity 改到 XML 再到 Service链式改动搞了十分钟。后来在 GitHub 上逛的时候看到了 MyBatis-Plus试着把它引入到自己的课设项目里发现明明能少写很多代码但班上同学用它的不多 —— 大部分人都觉得 反正 MyBatis 也能用干嘛多学一个。于是我把自己的课设从 MyBatis 迁到了 MyBatis-Plus过程踩了一堆坑但也实实在在少写了很多重复代码。这篇文章就是我的踩坑记录从一个正在学后端的普通大学生视角来写。读完你能收获什么每个改造点都知道原来怎么写、现在怎么写、坑在哪里拿到一套可以直接照着改的迁移顺序避开我踩过的那些让人调试到凌晨的坑不讲源码原理不讲官方文档里已有的内容只讲我自己亲历的东西。1. 依赖不是 加一个是 换一个我刚接触 MyBatis-Plus 的时候以为它是在 MyBatis 基础上再加一个依赖于是pom.xml里两个 starter 并存项目启动直接报 Bean 冲突看了半天日志才搞明白。原来 ——MyBatisdependency groupIdorg.mybatis.spring.boot/groupId artifactIdmybatis-spring-boot-starter/artifactId version2.3.0/version /dependency迁移后 ——MyBatis-Plusdependency groupIdcom.baomidou/groupId artifactIdmybatis-plus-boot-starter/artifactId version3.5.5/version /dependencyMyBatis-Plus 的 starter 已经内置了 MyBatis 和 MyBatis-Spring不需要也不应该再单独引入 MyBatis 依赖。如果你之前用了mybatis-spring-boot-starter直接删掉换上面这个。踩坑记录我第一次引入的时候只把 starter 换了但之前单独引的mybatis这个 artifact 忘了删。结果启动时报NoSuchMethodError查了半天 StackOverflow 才定位到是版本冲突。建议换完之后直接跑一下依赖树mvn dependency:tree | grep mybatis看到两个版本的 mybatis 相关包同时出现就说明还有没清干净的。2. Mapper 继承 BaseMapper—— 单表 CRUD 全消失了这是我感受到 MP 最有用的一步。原来在课设里 ——Mapper 接口 XML 双份维护Mapper public interface UserMapper { User selectById(Long id); int insert(User user); int updateById(User user); int deleteById(Long id); ListUser selectByCondition(String name, Integer age); }insert idinsert parameterTypecom.example.entity.User INSERT INTO user(name, age, email) VALUES(#{name}, #{age}, #{email}) /insert update idupdateById parameterTypecom.example.entity.User UPDATE user SET name#{name}, age#{age}, email#{email} WHERE id#{id} /update delete iddeleteById parameterTypelong DELETE FROM user WHERE id#{id} /delete select idselectById resultTypecom.example.entity.User SELECT * FROM user WHERE id#{id} /select课设里一个模块四五个这样的方法XML 轻轻松松上百行。这还是字段少的如果字段三十个光一个 insert 就几十行。迁移后 —— 继承 BaseMapperMapper public interface UserMapper extends BaseMapperUser { // 单表 CRUD 一个方法都不用写 }BaseMapper自带的方法包括方法说明insert(T entity)插入一条记录deleteById(Serializable id)根据 ID 删除updateById(T entity)根据 ID 更新selectById(Serializable id)根据 ID 查询selectList(WrapperT wrapper)条件查询列表selectPage(PageT page, WrapperT wrapper)分页查询selectCount(WrapperT wrapper)条件统计基本上原来单表的增删改查方法这些全包了。踩坑记录我第一次改的时候UserMapper继承了BaseMapper但忘了删 XML 里同名的方法。结果 BaseMapper 自带的insert方法调用时一直跳进我旧的 XML 里执行找了半天才发现 MyBatis 解析时 XML 会覆盖 BaseMapper 的同名 statement。迁移时记得把那些纯单表操作的 XML 片段删掉只保留多表联查。迁移前后对比迁移前 迁移后 ├── UserMapper.java (30行) ├── UserMapper.java (3行) ├── UserMapper.xml (200行) ├── UserMapper.xml (40行只剩联查) ├── OrderMapper.java (30行) ├── OrderMapper.java (3行) ├── OrderMapper.xml (180行) ├── OrderMapper.xml (35行只剩联查) └── ... └── ...3. XML 不用全删 —— 复杂查询原样保留迁移时我犯的第一个错误就是想把所有 XML 全删了全换 MP 的写法。后来发现根本没必要也做不到。MyBatis-Plus 对自定义 XML 完全兼容。Mapper 接口继承BaseMapper之后照样可以在同一个接口里定义自己的方法去调 XML。Mapper public interface UserMapper extends BaseMapperUser { // 多表联查——XML 写法和原来一模一样 ListUserOrderVO selectUserOrders(Param(userId) Long userId); // 复杂统计 BigDecimal selectTotalAmount(Param(startDate) Date startDate, Param(endDate) Date endDate); }对应的 XMLselect idselectUserOrders resultTypecom.example.vo.UserOrderVO SELECT u.name, o.order_no, o.amount FROM user u LEFT JOIN orders o ON u.id o.user_id WHERE u.id #{userId} /select一句话总结单表 CRUD 交给 BaseMapper多表联查和复杂 SQL 继续用 XML互不干扰。踩坑记录我当时自作聪明把 XML 里的resultMap映射全改成了实体类上的TableField注解结果 XML 引用不到映射直接报错。实际上你只需要在实体类上标注表名和主键XML 里的 resultMap 该怎么写还怎么写不用动。4. 条件构造器 ——WHERE 条件的写法完全不一样了这应该是我写代码时变化最大的一个点。原来课设里拼查询条件要么在 XML 里用if标签写动态 SQL要么在 Mapper 接口加一堆重载方法。MP 用条件构造器来替代。4.1 最基础QueryWrapper// 需求按姓名模糊搜索 年龄大于最小值 状态过滤按创建时间倒序 QueryWrapperUser wrapper new QueryWrapper(); wrapper.like(name, keyword) .gt(age, 18) .eq(status, 1) .orderByDesc(create_time); ListUser users userMapper.selectList(wrapper);等价于原来在 XML 里写的select idselectByCondition resultTypeUser SELECT * FROM user WHERE name LIKE CONCAT(%, #{keyword}, %) AND age 18 AND status 1 ORDER BY create_time DESC /select区别在于用 QueryWrapper 你不需要每多一个筛选条件就去改 XML 或 Mapper 接口Controller 里直接链式调用拼条件就行。4.2 推荐写法LambdaQueryWrapperQueryWrapper 有个问题name是手写字符串字段名写错了编译不报错运行才炸。// 不推荐字段名是字符串重构时 IDE 改不了这里 wrapper.eq(naem, value); // 拼错了编译器不会告诉你 // 推荐用 Lambda 引用编译期就能检查 LambdaQueryWrapperUser wrapper new LambdaQueryWrapper(); wrapper.like(User::getName, keyword) .gt(User::getAge, 18) .eq(User::getStatus, 1) .orderByDesc(User::getCreateTime);以后实体类字段重命名IDE 一键重构这里自动跟着改再也不怕手滑打错字了。4.3 常用条件方法速查方法SQL 等价说明eq(name, val)name val等于ne(name, val)name val不等于gt(age, 18)age 18大于ge(age, 18)age 18大于等于lt(age, 18)age 18小于le(age, 18)age 18小于等于like(name, val)name LIKE %val%全模糊性能差慎用likeLeft(name, val)name LIKE %val左模糊likeRight(name, val)name LIKE val%右模糊走索引in(id, list)id IN (1,2,3)集合查询between(age, 18, 60)age BETWEEN 18 AND 60区间isNull(email)email IS NULL判空orderByAsc(age)ORDER BY age ASC升序orderByDesc(age)ORDER BY age DESC降序last(LIMIT 1)拼在最后慎用有 SQL 注入风险4.4 动态条件 —— 不用写if标签了原来在 XML 里拼动态条件select idselectByCondition resultTypeUser SELECT * FROM user WHERE 11 if testname ! null and name ! AND name LIKE CONCAT(%, #{name}, %) /if if testage ! null AND age #{age} /if /select用 Wrapper 的eq、like等方法传null时自动忽略该条件LambdaQueryWrapperUser wrapper new LambdaQueryWrapper(); wrapper.like(StringUtils.hasText(name), User::getName, name) .gt(age ! null, User::getAge, age) .eq(status ! null, User::getStatus, status);第一个参数是boolean condition为false时不拼接该条件。这个设计让动态条件一行一个比 XML 的if堆叠清爽太多了。踩坑 1likevslikeRight。全模糊LIKE %keyword%不走索引能用likeRight前缀匹配LIKE keyword%就别用like。虽说课设数据量小感觉不出来但养成习惯很重要。踩坑 2别在last()里拼用户输入的内容。last()是直接字符串拼接不做参数化处理有 SQL 注入风险。只用来写硬编码的片段比如last(LIMIT 1)。5. 分页 —— 从手写两条 SQL 到一行 page ()课设里写列表接口套路基本是先写一条SELECT COUNT(*)查总数再写一条SELECT ... LIMIT offset, size查数据然后手动算offset (pageNum - 1) * pageSize。每张表都要重复一遍写得手酸。5.1 先配分页插件MP 的分页需要注册一个拦截器这一步我第一次漏掉了后面踩了个大坑Configuration public class MybatisPlusConfig { Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { MybatisPlusInterceptor interceptor new MybatisPlusInterceptor(); PaginationInnerInterceptor paginationInterceptor new PaginationInnerInterceptor(); paginationInterceptor.setDbType(DbType.MYSQL); // 按你的数据库类型来 interceptor.addInnerInterceptor(paginationInterceptor); return interceptor; } }MySQL 用DbType.MYSQLPostgreSQL 是DbType.POSTGRE_SQL。5.2 使用分页// 构建分页对象 PageUser page new Page(pageNum, pageSize); // 查询——自动查总数自动加 LIMIT PageUser result userMapper.selectPage(page, wrapper); // 取结果 ListUser records result.getRecords(); // 当前页数据 long total result.getTotal(); // 总记录数 long pages result.getPages(); // 总页数原来二三十行代码加两段 XML现在三行搞定。自定义 SQL 也能分页在 Mapper 方法里传入Page参数Mapper public interface UserMapper extends BaseMapperUser { PageUserOrderVO selectUserOrders(PageUserOrderVO page, Param(userId) Long userId); }select idselectUserOrders resultTypecom.example.vo.UserOrderVO SELECT u.name, o.order_no, o.amount FROM user u LEFT JOIN orders o ON u.id o.user_id WHERE u.id #{userId} /selectXML 里不用写 LIMIT 和 COUNTMP 自动帮你加。踩坑记录我第一次配的时候忘了写MybatisPlusInterceptor这个 Bean然后分页死活不生效查出来的数据永远是全表。课设里数据才几十条没发现后来往数据库里导了 5 万条测试数据接口直接卡了十几秒才反应过来。加上分页插件后一定验证一下执行的 SQL 有没有带 LIMIT。6. 主键策略 —— 不用每次都手动 set ID 了原来// 方式一依赖数据库自增 userMapper.insert(user); Long id user.getId(); // 依赖 MyBatis 的 useGeneratedKeys // 方式二自己生成雪花 ID user.setId(IdWorker.getId()); userMapper.insert(user);迁移后 —— 一个注解搞定Data TableName(user) public class User { TableId(type IdType.ASSIGN_ID) private Long id; private String name; }insert时不用手动 set IDMP 自动填充。几种主键策略策略说明适用场景ASSIGN_ID默认雪花算法Long 型分布式系统数据量大ASSIGN_UUIDUUIDString 型不需要有序 IDAUTO数据库自增单库MySQL 自增主键INPUT手动赋值使用外部生成的 ID全局改默认策略mybatis-plus: global-config: db-config: id-type: auto踩坑记录雪花算法生成的 ID 是 19 位的 Long超过了 JavaScript 的Number.MAX_SAFE_INTEGER2^53。我课设后端返回的 JSON 里 ID 是1234567890123456789前端拿到的变成了1234567890123456700精度丢了。前端同学调了半天说 接口返回的 ID 和数据库对不上查了俩小时才发现是 JSON 序列化的问题。解决方案—— 全局配置 Jackson 把 Long 转成 StringConfiguration public class JacksonConfig { Bean public Jackson2ObjectMapperBuilderCustomizer jackson2ObjectMapperBuilderCustomizer() { return builder - { builder.serializerByType(Long.class, ToStringSerializer.instance); builder.serializerByType(Long.TYPE, ToStringSerializer.instance); }; } }或者直接在字段上加TableId(type IdType.ASSIGN_ID) JsonSerialize(using ToStringSerializer.class) private Long id;7. 自动填充 ——createTime/updateTime 不用到处写了课设里每次 insert 和 update 都要手动 set 时间// 插入时 user.setCreateTime(new Date()); user.setUpdateTime(new Date()); userMapper.insert(user); // 更新时 user.setUpdateTime(new Date()); userMapper.updateById(user);代码写多了经常忘掉某一处然后数据库里的时间字段就是 null。查 Bug 的时候不知道这条记录是什么时候创建的很头疼。迁移后 —— 写一次全局生效Component public class MyMetaObjectHandler implements MetaObjectHandler { Override public void insertFill(MetaObject metaObject) { this.strictInsertFill(metaObject, createTime, Date.class, new Date()); this.strictInsertFill(metaObject, updateTime, Date.class, new Date()); } Override public void updateFill(MetaObject metaObject) { this.strictUpdateFill(metaObject, updateTime, Date.class, new Date()); } }实体类字段上标注Data public class User { TableField(fill FieldFill.INSERT) private Date createTime; TableField(fill FieldFill.INSERT_UPDATE) private Date updateTime; }FieldFill有四种值触发时机INSERT插入时填充UPDATE更新时填充INSERT_UPDATE插入和更新都填充DEFAULT不自动处理踩坑记录strictInsertFill是严格模式 —— 字段已经有值的时候不会覆盖。反过来setFieldValByName是无脑覆盖不管你手动设没设。我一开始两个混用排查了很久的 为什么我设的时间没有生效。建议统一用strict系列方法。8. 乐观锁 —— 一个 Version 注解不用手写版本号判断了乐观锁的原理每次更新时检查版本号对上了才更新同时版本号 1对不上说明别人先改了更新失败。原来手写步骤多还容易漏// 1. 查出当前版本 User user userMapper.selectById(id); Integer currentVersion user.getVersion(); // 2. 更新时带上版本号 user.setBalance(user.getBalance().subtract(amount)); user.setVersion(currentVersion 1); // 3. XML 里写UPDATE user SET balance#{balance}, versionversion1 // WHERE id#{id} AND version#{version} // 4. 检查影响行数——漏了这一步就白搭 int rows userMapper.updateByIdWithVersion(user); if (rows 0) { throw new RuntimeException(数据已被修改请重试); }四个步骤环环相扣漏一个乐观锁就形同虚设了。我在课设的秒杀模块里漏了第 4 步测试时超卖了一单还好只是模拟数据。迁移后—— 实体类上加个注解就行Data public class Product { private Long id; private String name; private BigDecimal price; private Integer stock; Version private Integer version; }配置拦截器Bean public MybatisPlusInterceptor mybatisPlusInterceptor() { MybatisPlusInterceptor interceptor new MybatisPlusInterceptor(); interceptor.addInnerInterceptor(new OptimisticLockerInnerInterceptor()); // 别忘了分页拦截器 interceptor.addInnerInterceptor(new PaginationInnerInterceptor(DbType.MYSQL)); return interceptor; }使用时就正常更新Product product productMapper.selectById(1L); product.setStock(product.getStock() - 10); int rows productMapper.updateById(product); // 自动执行UPDATE product SET stock?, versionversion1 // WHERE id? AND version? // rows0 说明版本冲突 if (rows 0) { throw new BusinessException(库存已变化请刷新后重试); }踩坑 1别忘了配OptimisticLockerInnerInterceptor否则Version不会生效SQL 里不会带 version 条件也不会自增。踩坑 2updateById如果传进来一个只设了 id 和要改字段的新对象version 是 nullMP 就不会带 version 条件。想用乐观锁保护更新必须先查出完整对象修改后再调 updateById。9. 逻辑删除 —— 不用每张表都手写软删除了原来做软删除每个删除接口写UPDATE user SET is_deleted 1 WHERE id ?每个查询的 SQL 都要带AND is_deleted 0。忘了带这个条件用户就能在列表里看到 已删除 的记录。迁移后——yaml 配一下加个注解就搞定mybatis-plus: global-config: db-config: logic-delete-field: deleted # 逻辑删除字段名 logic-delete-value: 1 # 已删除的值 logic-not-delete-value: 0 # 未删除的值实体类Data public class User { private Long id; TableLogic private Integer deleted; }效果// 你写的是 userMapper.deleteById(1L); // 实际执行UPDATE user SET deleted 1 WHERE id 1 AND deleted 0 // 你写的是 ListUser users userMapper.selectList(null); // 实际执行SELECT * FROM user WHERE deleted 0所有 SELECT 自动带deleted 0所有 DELETE 自动变 UPDATE非常省心。踩坑 1逻辑删除 唯一索引。比如手机号有唯一索引一个用户注销逻辑删除后另一个用户想用同一个手机号注册会因唯一索引冲突失败。解决方案要么把 deleted 字段加入联合唯一索引要么注销时把原值改成deleted_时间戳_原值。踩坑 2手写 XML 里的查询不会自动加deleted 0。如果自定义 SQL 也要过滤逻辑删除的数据得自己加AND deleted 0。10. 别急着用代码生成器先手改两个 MapperMP 的代码生成器AutoGenerator确实很爽 —— 连上数据库一键生成 Entity、Mapper、Service、Controller 全套代码分分钟搞定一个模块。但我的建议是初学的时候先别用它。原因很朴素生成出来的代码你不知道 BaseMapper 到底替你干了什么ServiceImpl 替你干了什么。跑起来没问题还好一旦有问题三个层面生成器代码、MP 框架、自己的业务逻辑搅在一起根本定位不到根因。我自己后来摸索出的顺序第一步改依赖把 mybatis-spring-boot-starter 换成 mybatis-plus-boot-starter 第二步挑 2 张最简单的表手改 Mapper 继承 BaseMapper删对应的 XML 单表操作 第三步在这 2 张表上把分页、自动填充、乐观锁、逻辑删除全配好确认跑通 第四步搞明白每个特性到底帮你干了什么之后再上代码生成器迁移剩下的表我第一次就是一脸懵直接上生成器结果分页、乐观锁、逻辑删除全是生成出来的默认配置好多都没真正启用但自己不知道。后来花了两天一行一行对比代码还不如一开始手写。总结这篇文章没讲 MyBatis-Plus 的全部功能挑了我在课设迁移过程中收益最高、自己也踩过坑的 10 个点。从一个普通大学生的角度来说MyBatis-Plus 最大的好处不是它功能多强而是它帮你省掉了大量 重复但不写不行 的代码。课设里几个模块改完之后XML 少了一大半终于不用在 加字段 → 改 XML → 改 Mapper → 改 Service 的链式操作上折腾了。如果让我用一句话总结MyBatis-Plus 不是要替代 MyBatis是让你把精力从写单表 CRUD 转移到搞定业务逻辑上。你也在用 MyBatis-Plus 吗踩过什么不一样的坑或者你还在犹豫要不要学顾虑是什么评论区聊聊吧。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2633933.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

SpringBoot-17-MyBatis动态SQL标签之常用标签

文章目录 1 代码1.1 实体User.java1.2 接口UserMapper.java1.3 映射UserMapper.xml1.3.1 标签if1.3.2 标签if和where1.3.3 标签choose和when和otherwise1.4 UserController.java2 常用动态SQL标签2.1 标签set2.1.1 UserMapper.java2.1.2 UserMapper.xml2.1.3 UserController.ja…

wordpress后台更新后 前端没变化的解决方法

使用siteground主机的wordpress网站,会出现更新了网站内容和修改了php模板文件、js文件、css文件、图片文件后,网站没有变化的情况。 不熟悉siteground主机的新手,遇到这个问题,就很抓狂,明明是哪都没操作错误&#x…

网络编程(Modbus进阶)

思维导图 Modbus RTU(先学一点理论) 概念 Modbus RTU 是工业自动化领域 最广泛应用的串行通信协议,由 Modicon 公司(现施耐德电气)于 1979 年推出。它以 高效率、强健性、易实现的特点成为工业控制系统的通信标准。 包…

UE5 学习系列(二)用户操作界面及介绍

这篇博客是 UE5 学习系列博客的第二篇,在第一篇的基础上展开这篇内容。博客参考的 B 站视频资料和第一篇的链接如下: 【Note】:如果你已经完成安装等操作,可以只执行第一篇博客中 2. 新建一个空白游戏项目 章节操作,重…

IDEA运行Tomcat出现乱码问题解决汇总

最近正值期末周,有很多同学在写期末Java web作业时,运行tomcat出现乱码问题,经过多次解决与研究,我做了如下整理: 原因: IDEA本身编码与tomcat的编码与Windows编码不同导致,Windows 系统控制台…

利用最小二乘法找圆心和半径

#include <iostream> #include <vector> #include <cmath> #include <Eigen/Dense> // 需安装Eigen库用于矩阵运算 // 定义点结构 struct Point { double x, y; Point(double x_, double y_) : x(x_), y(y_) {} }; // 最小二乘法求圆心和半径 …

使用docker在3台服务器上搭建基于redis 6.x的一主两从三台均是哨兵模式

一、环境及版本说明 如果服务器已经安装了docker,则忽略此步骤,如果没有安装,则可以按照一下方式安装: 1. 在线安装(有互联网环境): 请看我这篇文章 传送阵>> 点我查看 2. 离线安装(内网环境):请看我这篇文章 传送阵>> 点我查看 说明&#xff1a;假设每台服务器已…

XML Group端口详解

在XML数据映射过程中&#xff0c;经常需要对数据进行分组聚合操作。例如&#xff0c;当处理包含多个物料明细的XML文件时&#xff0c;可能需要将相同物料号的明细归为一组&#xff0c;或对相同物料号的数量进行求和计算。传统实现方式通常需要编写脚本代码&#xff0c;增加了开…

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器的上位机配置操作说明

LBE-LEX系列工业语音播放器|预警播报器|喇叭蜂鸣器专为工业环境精心打造&#xff0c;完美适配AGV和无人叉车。同时&#xff0c;集成以太网与语音合成技术&#xff0c;为各类高级系统&#xff08;如MES、调度系统、库位管理、立库等&#xff09;提供高效便捷的语音交互体验。 L…

(LeetCode 每日一题) 3442. 奇偶频次间的最大差值 I (哈希、字符串)

题目&#xff1a;3442. 奇偶频次间的最大差值 I 思路 &#xff1a;哈希&#xff0c;时间复杂度0(n)。 用哈希表来记录每个字符串中字符的分布情况&#xff0c;哈希表这里用数组即可实现。 C版本&#xff1a; class Solution { public:int maxDifference(string s) {int a[26]…

【大模型RAG】拍照搜题技术架构速览:三层管道、两级检索、兜底大模型

摘要 拍照搜题系统采用“三层管道&#xff08;多模态 OCR → 语义检索 → 答案渲染&#xff09;、两级检索&#xff08;倒排 BM25 向量 HNSW&#xff09;并以大语言模型兜底”的整体框架&#xff1a; 多模态 OCR 层 将题目图片经过超分、去噪、倾斜校正后&#xff0c;分别用…

【Axure高保真原型】引导弹窗

今天和大家中分享引导弹窗的原型模板&#xff0c;载入页面后&#xff0c;会显示引导弹窗&#xff0c;适用于引导用户使用页面&#xff0c;点击完成后&#xff0c;会显示下一个引导弹窗&#xff0c;直至最后一个引导弹窗完成后进入首页。具体效果可以点击下方视频观看或打开下方…

接口测试中缓存处理策略

在接口测试中&#xff0c;缓存处理策略是一个关键环节&#xff0c;直接影响测试结果的准确性和可靠性。合理的缓存处理策略能够确保测试环境的一致性&#xff0c;避免因缓存数据导致的测试偏差。以下是接口测试中常见的缓存处理策略及其详细说明&#xff1a; 一、缓存处理的核…

龙虎榜——20250610

上证指数放量收阴线&#xff0c;个股多数下跌&#xff0c;盘中受消息影响大幅波动。 深证指数放量收阴线形成顶分型&#xff0c;指数短线有调整的需求&#xff0c;大概需要一两天。 2025年6月10日龙虎榜行业方向分析 1. 金融科技 代表标的&#xff1a;御银股份、雄帝科技 驱动…

观成科技:隐蔽隧道工具Ligolo-ng加密流量分析

1.工具介绍 Ligolo-ng是一款由go编写的高效隧道工具&#xff0c;该工具基于TUN接口实现其功能&#xff0c;利用反向TCP/TLS连接建立一条隐蔽的通信信道&#xff0c;支持使用Let’s Encrypt自动生成证书。Ligolo-ng的通信隐蔽性体现在其支持多种连接方式&#xff0c;适应复杂网…

铭豹扩展坞 USB转网口 突然无法识别解决方法

当 USB 转网口扩展坞在一台笔记本上无法识别,但在其他电脑上正常工作时,问题通常出在笔记本自身或其与扩展坞的兼容性上。以下是系统化的定位思路和排查步骤,帮助你快速找到故障原因: 背景: 一个M-pard(铭豹)扩展坞的网卡突然无法识别了,扩展出来的三个USB接口正常。…

未来机器人的大脑:如何用神经网络模拟器实现更智能的决策?

编辑&#xff1a;陈萍萍的公主一点人工一点智能 未来机器人的大脑&#xff1a;如何用神经网络模拟器实现更智能的决策&#xff1f;RWM通过双自回归机制有效解决了复合误差、部分可观测性和随机动力学等关键挑战&#xff0c;在不依赖领域特定归纳偏见的条件下实现了卓越的预测准…

Linux应用开发之网络套接字编程(实例篇)

服务端与客户端单连接 服务端代码 #include <sys/socket.h> #include <sys/types.h> #include <netinet/in.h> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/inet.h> #include <pthread.h> …

华为云AI开发平台ModelArts

华为云ModelArts&#xff1a;重塑AI开发流程的“智能引擎”与“创新加速器”&#xff01; 在人工智能浪潮席卷全球的2025年&#xff0c;企业拥抱AI的意愿空前高涨&#xff0c;但技术门槛高、流程复杂、资源投入巨大的现实&#xff0c;却让许多创新构想止步于实验室。数据科学家…

深度学习在微纳光子学中的应用

深度学习在微纳光子学中的主要应用方向 深度学习与微纳光子学的结合主要集中在以下几个方向&#xff1a; 逆向设计 通过神经网络快速预测微纳结构的光学响应&#xff0c;替代传统耗时的数值模拟方法。例如设计超表面、光子晶体等结构。 特征提取与优化 从复杂的光学数据中自…