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

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

作者逆境不可逃技术永无止境希望我的内容可以帮助到你本节目属于专栏《后端新手谈》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 吗踩过什么不一样的坑或者你还在犹豫要不要学顾虑是什么评论区聊聊吧。

相关文章:

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

作者:逆境不可逃 技术永无止境 希望我的内容可以帮助到你!!!!! 本节目属于专栏《后端新手谈》:https://blog.csdn.net/2401_87662859/category_13141790.html 大家吼 ! 我是 逆境不可逃 今天给…...

OpenAI通用推理模型攻克80年数学难题,跨领域推理能力引发科学研究范式变革!

极其简单的谜题,与阻挡人类80年的高墙要理解这项突破有多么不可思议,我们必须先回到1946年。那一年,20世纪最伟大的传奇数学家之一保罗埃尔德什(Paul Erdős)提出了一个几何问题:如果在二维平面上任意画下n…...

Mardi 品牌创始人是谁?一文读懂法国 Mardi Ladin

法国 Mardi Ladin 品牌创始人是La Bergon(Baudino Cd L),一位出身法国时尚世家的设计师,品牌的灵感直接来自于 1975 年法国经典电影《表兄妹》中入围奥斯卡最佳女主角的角色 "玛尔蒂 MARDI"。创始人 La Bergon 解析La B…...

2026年,IP地理位置精准查询的几个硬核技术变化

关于IP定位相关最近和几个同行交流,发现大家对IP定位的理解还停留在之前,想把自己这段时间的一些实践整理出来,希望能给同样在搞网络或风控的同行一些参考。 IPv6流量超过IPv4、住宅代理攻击泛滥、CGNAT覆盖越来越广……这些变化正在悄悄改变…...

python 内存管理 内存泄漏及排查方案 内存友好的python代码

Python 内存管理 一、一句话总结 Python 的内存管理就是三件事: 自动分配内存(你不用管变量存在哪)自动回收垃圾(不用的对象自动删掉)靠引用计数 分代垃圾回收实现二、核心机制 1:引用计数(最基…...

解锁.NET 11 新境:ASP.NET Core 10 在微服务安全通信的深化与实践

解锁.NET 11 新境:ASP.NET Core 10 在微服务安全通信的深化与实践 前言 在当今分布式系统盛行的时代,微服务架构已成为构建大型应用的主流选择。ASP.NET Core 10 作为.NET 11 生态中重要的后端框架,为微服务间的安全通信提供了全面且强大的支…...

为什么你的ElevenLabs马来文输出总像“机器人朗读”?资深语音架构师拆解4层韵律建模断层与3个修复级prompt模板

更多请点击: https://intelliparadigm.com 第一章:为什么你的ElevenLabs马来文输出总像“机器人朗读”?资深语音架构师拆解4层韵律建模断层与3个修复级prompt模板 马来语(Bahasa Melayu)虽属声调中性语言,…...

【AI入门知识点】Skills 是什么?终于有人把 Skills、Function Calling、MCP 讲明白了

为什么现在 AI 会查天气?为什么 AI 能读 Excel、操作浏览器、发邮件?为什么很多人说:未来 AI 拼的不是谁更聪明,而是谁 Skills 更多?很多刚学 AI 的人。都会被几个词搞晕:SkillsFunction CallingMCP看起来都…...

C++内存对齐与布局优化

C内存对齐与布局优化内存对齐是编译器为了提高内存访问效率而采用的策略。理解内存对齐规则对于优化结构体大小和提高程序性能至关重要。结构体的内存布局受对齐规则影响,可能包含填充字节。#include #includestruct Unaligned { char a; int b; char c; };struct A…...

C++内联函数性能分析

C内联函数性能分析内联函数通过在调用点展开函数体来消除函数调用开销。理解内联机制和使用场景对于编写高性能代码至关重要。inline关键字建议编译器内联函数。#include #includeinline int add(int a, int b) { return a b; }inline int multiply(int a, int b) { return a …...

设计模式之建造者

问题:构造函数参数太多(「伸缩构造」),或步骤必须按顺序、且步骤组合多变。做法:Director(可选)规定步骤顺序;Builder 提供 setA()、setB()… 最后 build() 返回产品。C 要点&#x…...

向日葵远程控制16.5发布,“免密远控”功能登场便捷又安全

人在公司,急需处理家里电脑上的重要文件,却完全想不起访问密码或者系统的帐号密码;出差在外,想远程操作办公室电脑,却不得不打电话让同事帮忙看一眼密码设置甚至干脆让同事点个接受......密码虽然是一种非常主流的安全…...

WTEW的操作记录

WTEW的操作记录WTEW事务代码的操作记录WTEW事务代码的操作记录 1、查询贸易合同信息 如果是自己创建可以使用WB21、WB22、WB23事务码,如果是税码更新用WBRP更新价格 2、创建后续单据,采购TC创建采购订单,销售TC创建销售订单,注…...

Google三星AI眼镜来了,开发者该关注什么

AI 眼镜又回来了,但这次不只是换个硬件外壳AI 眼镜这个话题,最近又被推到了台前。Google 在 I/O 2026 展示了基于 Android XR 的智能眼镜方向,并把三星、Gentle Monster、Warby Parker 等合作方一起摆上台面。按照目前公布的信息,…...

数据结构——带懒标记的线段树

一、什么是线段树?线段树是一种二叉树数据结构,用于高效地处理区间查询和区间更新操作。核心思想:将数组分成若干个区间(线段),每个节点代表一个区间,通过合并子节点的信息来得到父节点的信息。…...

2026年企业AI落地新趋势!RAG知识库实战指南:环境搭建到生产部署全解析

本文介绍了RAG(检索增强生成)技术在企业知识库中的应用,通过从环境搭建到生产部署的完整实战指南,阐述如何利用RAG提升大语言模型回答的准确性、可追溯性和时效性。文章涵盖了基础环境配置、技术选型、数据准备、知识库构建、RAG系…...

终极Mac微信插件:消息防撤回与多开登录完整指南

终极Mac微信插件:消息防撤回与多开登录完整指南 【免费下载链接】WeChatExtension-ForMac A plugin for Mac WeChat 项目地址: https://gitcode.com/gh_mirrors/we/WeChatExtension-ForMac 还在为Mac微信无法防撤回消息而烦恼吗?想要在同一台电脑…...

一文讲清WMS软件是什么?企业为什么要用WMS软件?

在数字化供应链时代,WMS软件(仓储管理系统)已成为企业物流管理的核心。面对仓库混乱、库存不准,很多企业都在问:WMS软件到底是什么?它和Excel或进销存有什么区别?企业为什么要用WMS软件&#xf…...

Java基础小知识

一、 计算机基础知识1.计算机硬件的分类:运算器 控制器 存储器 输入设备 输出设备二、cmd命令窗口的基本用法操着: 说明:盘符名称 : 盘符切换。E:回车,表示切换到E盘dir 查看当前路径下的内容cd 目录 进入单级目录。cd…...

十三张扑克APP

能开发十三张扑克APP的请联系我,有客户渠道需要这类APP,要开发很多款十三张...

P2-CIFAR彩色图片识别

● 🍨 本文为🔗365天深度学习训练营中的学习记录博客 ● 🍖 原作者:K同学啊学习目标:1.编写一个完整的深度学习程序 2. 手动推导卷积层与池化层的计算过程一、前期准备1.设置GPUimport torch import torch.nn as nn im…...

CANN 算子融合技术:Conv-BN-ReLU 与 MatMul-LayerNorm 等融合模式深度解析

CANN 算子融合技术:Conv-BN-ReLU 与 MatMul-LayerNorm 等融合模式深度解析算子融合是提升性能的关键手段。本文深入讲解昇腾支持的算子融合技术、实现原理和应用实践。一、融合技术概述 1.1 为什么要融合 原始: Conv → BN → ReLU → Conv → BN → ReLU融合前内存…...

Gitea库完整从Ubuntu迁移到CentOS中

文章目录 一、概述 二、数据迁移 2.1 获取数据存储路径 2.2 搞事之前先备份(目标服务器CentOS) 2.2.1 停止gitea服务 2.2.2 备份gitea文件夹 2.3 从Ubuntu的数据目录中将数据拷贝到CentOS中 2.4 备份mysql数据库并拷贝到目标服务器(CentOS) 2.4.1 通过mysqldump备份数据库 …...

复杂干扰下考虑异质性的非机动车微观行为建模与仿真【附仿真】

✨ 长期致力于非机动车微观交通行为、异质性、感知—决策—行动三阶段、社会力模型、模糊逻辑研究工作,擅长数据搜集与处理、建模仿真、程序编写、仿真设计。 ✅ 专业定制毕设、代码 ✅ 如需沟通交流,点击《获取方式》 (1)非机动车…...

(二) 1. Q-learning的遗憾界分析-高效的Q-learning算法

高效的Q-learning算法 1.1. 无模型算法 1.2. UCB算法 1.3. 文献回顾 无模型(Model-free)强化学习算法(如 Q-learning)无需显式地对环境进行建模,而是直接对价值函数或策略进行参数化和更新。与基于模型(Model-based)的方法相比,这类算法通常更简单、更灵活,因此在现代…...

企业微信外部群如何通过 API 自动化投递结构化小程序卡片

能力介绍 相比于传统的文字链接,结构化的小程序卡片拥有更高的点击率和更规范的视觉展现。该能力允许开发者通过主动调用 API,直接向指定的企业微信外部群投递原生小程序卡片。接口支持自定义动态配置小程序的 appid、首屏页面路径 pagepath&#xff08…...

obsidian博客联动方案

平台文章具有滞后性,最新文章请访问https://blog.nuoyis.net 原先博客需要使用typorapicgotypecho,其中typora编写完毕后需要复制到typecho后台去,极其不方便,然后经过高人指点,我对该软件交互使用开发了新高度 obsidi…...

【考研】2026/5/21

政治2026/5/21唯物辩证法本质上是批判的和革命的:在唯物辩证法看来,一切事物都处在发生、发展和灭亡的过程中,“不存在任何最终的东西、绝对的东西、神圣的东西”。唯物辩证法是客观辩证法与主观辩证法的统一:①客观辩证法&#x…...

1987年4月26日下午15-17点出生性格、运势和命运

1987年4月24日晚上出生的人,如今已步入38岁的门槛。在职业生涯中,这是一个承上启下的关键阶段——既脱离了职场新人的青涩,又尚未到达管理者或专家的巅峰位置。从非命理的角度分析,他们的事业运势与时代变迁、个人选择和社会结构密…...

企业AI合规:数据安全生死线

企业大模型应用中的数据安全合规体系建设 前言:数据安全合规——企业AI落地的必答题 一、合规风险识别与关键挑战 二、技术架构设计与安全合规方案 针对上述四大风险挑战,企业需要从技术架构层面构建纵深防御体系。以下从数据脱敏、访问控制、日志审计、…...