发布时间:2026/7/3 22:00:44
从源码到代码:MyBatis-Flex 与 MyBatis-Plus 的逐项对比
社区里好多程序员在讨论MyBatis-Flex说它轻量、快、设计优雅。好奇心驱动下载了源码搭建了一个demo工程认认真真学了一遍。过程中发现它和MyBatis-Plus的设计思路差异不小记录下来做个对比。这篇文章不评价谁好谁差只是从源码和实际代码两个层面看看这两个框架到底有什么不同。demo工程用的订单表和订单明细表Spring Boot 2.7MyBatis-Flex 1.11.8。同一张订单表两种写法先来段最简单的代码对同一张表两个框架的代码长什么样。实体类MyBatis-Flex的实体类用Table和Id注解Table(order)DatapublicclassOrder{Id(keyTypeKeyType.Auto)privateLongid;privateStringorderNo;privateLonguserId;privateBigDecimaltotalAmount;privateIntegerstatus;privateLocalDateTimecreateTime;}MyBatis-Plus的写法大家应该都很熟悉了TableName和TableIdTableName(order)DatapublicclassOrder{TableId(typeIdType.AUTO)privateLongid;privateStringorderNo;privateLonguserId;privateBigDecimaltotalAmount;privateIntegerstatus;privateLocalDateTimecreateTime;}注解名不同但做的事情一样。真正的差异在查询条件的构建方式上。条件查询查某个用户的所有已支付订单按创建时间倒序。MyBatis-Plus用LambdaQueryWrapperLambdaQueryWrapperOrderwrappernewLambdaQueryWrapper();wrapper.eq(Order::getUserId,userId).eq(Order::getStatus,1).orderByDesc(Order::getCreateTime);ListOrderordersorderMapper.selectList(wrapper);MyBatis-Flex用QueryWrapper但条件构建方式完全不同QueryWrapperqueryQueryWrapper.create().where(ORDER.USER_ID.eq(userId)).and(ORDER.STATUS.eq(1)).orderBy(ORDER.CREATE_TIME.desc());ListOrderordersorderMapper.selectListByQuery(query);注意这里的ORDER不是字符串是一个编译期自动生成的类。ORDER.USER_ID、ORDER.STATUS都是这个类里的常量字段。写错了字段名编译直接报错不需要等到运行时才发现。这个ORDER类是怎么来的后面讲架构差异的时候会详细说。分页查询MyBatis-Plus的分页需要先配置拦截器然后创建Page对象// 需要先配置 MybatisPlusInterceptor PaginationInnerInterceptorPageOrderpagenewPage(1,10);orderMapper.selectPage(page,wrapper);MyBatis-Flex的分页是内建的不需要配置拦截器直接调paginate()PageOrderpageorderMapper.paginate(1,10,query);看起来只是一个方法调用的区别背后是两个框架在架构设计上的根本分歧。架构上的根本差异写法不同只是表象在设计层面上看它们生成SQL的方式是不同的。MyBatis-Plus启动期注入MyBatis-Plus在MyBatis启动阶段通过AbstractSqlInjector为每个Mapper接口注入CRUD对应的MappedStatement。每个操作背后都有一个专门的类来负责拼SQLInsert类负责插入DeleteById类负责按ID删除SelectList类负责列表查询等等。这些类都是AbstractMethod的子类。启动时AbstractSqlInjector遍历所有注册的AbstractMethod为每个Mapper逐个注入。MyBatis-Plus是在MyBatis原有的XML解析机制之上做扩展。它把CRUD操作的SQL模板预编译好注册到MyBatis的Configuration里运行时直接拿来用。这套机制的代价是每个Mapper接口不管你用不用启动时都会注入一整套CRUD方法。MyBatis-FlexProvider注解MyBatis-Flex走了一条完全不同的路。它的BaseMapper上的方法用的是MyBatis原生的SelectProvider、InsertProvider等注解指向一个EntitySqlProvider类。SQL不是在启动时预生成的而是在运行时由Provider动态拼出来的。调用orderMapper.selectOneById(1)的时候MyBatis会调用EntitySqlProvider里对应的方法这个方法根据实体类的元数据表名、字段、主键等实时拼出一条SQL。这种设计带来了MyBatis-Flex官网一直在强调的「三个轻」轻依赖整个框架只依赖MyBatis没有其他任何第三方依赖。轻实现没有拦截器。MyBatis-Plus的分页、租户、乐观锁等功能都是通过拦截器实现的MyBatis-Flex把这些能力直接内建在core里不走拦截器。轻运行没有SQL解析。MyBatis-Plus的拦截器在执行前会解析原始SQL比如分页拦截器要解析SQL来生成count语句MyBatis-Flex直接拼SQL不需要解析。APT编译期代码生成前面条件查询里用到的ORDER类不是手写的是编译期自动生成的。MyBatis-Flex用了一个叫APTAnnotation Processing Tool的技术和Lombok的原理类似。在mvn compile的时候mybatis-flex-processor模块会扫描所有带Table注解的实体类自动生成两样东西一个是TableDef类比如OrderTableDef里面包含每个字段对应的QueryColumn常量。ORDER.USER_ID就是OrderTableDef里的一个QueryColumn它知道这个字段对应哪张表的哪一列。另一个是Mapper接口。如果项目里没有手写MapperAPT会自动生成一个继承BaseMapper的接口。这套机制的好处是查询条件的构建是类型安全的。ORDER.USER_NAME假设有这个字段写错了字段名IDE直接标红编译都过不了。MyBatis-Plus的Lambda方式也能做到编译期检查但它依赖实体类的getter方法Flex这边不需要直接引用字段常量就行。不过这个设计也有代价用MyBatis-Flex写查询你得知道两个类——Order实体和ORDERAPT生成的TableDef。新人刚接触的时候可能会懵这个ORDER是哪来的它在源码里看不到是编译后才会出现的类。而用MyBatis-Plus只需要知道Order一个类就够了Order::getUserId这种方法引用很直观不需要理解额外的生成机制。社区里讨论框架选型的时候不少人提到MyBatis-Flex的学习曲线比Plus陡APT生成的这些类就是原因之一。那这个取舍值不值Flex用「多一个类」换来的是不依赖getter方法、支持多表join、QueryWrapper可序列化传输。这些都是Plus的Lambda方式做不到的。但如果你只是做单表CRUDPlus的方式确实更简单直接。多表查询差异最大的地方单表CRUD两个框架差别不大真正与众不同的是多表查询。假设要查询已支付订单及其明细在MyBatis-Plus里QueryWrapper不支持join你得手写XMLselectidlistWithDetailresultTypeOrderSELECT o.*, d.product_name, d.price, d.quantity FROM order o LEFT JOIN order_detail d ON o.id d.order_id WHERE o.status 1/selectMapper接口里还得加一个对应的方法声明。代码量不多但每个多表查询都得这么写一遍。在MyBatis-Flex里QueryWrapper直接支持leftJoinQueryWrapperqueryQueryWrapper.create().select().from(ORDER).leftJoin(ORDER_DETAIL).on(ORDER.ID.eq(ORDER_DETAIL.ORDER_ID)).where(ORDER.STATUS.eq(1));ListOrderordersorderMapper.selectListByQuery(query);不需要写XML不需要额外声明Mapper方法。join条件用的是编译期生成的QueryColumn字段名写错了编译就报错。这个差异在项目里影响很大。用过MyBatis-Plus的人都知道稍微复杂一点的查询最终都得回到XMLQueryWrapper能覆盖的场景其实有限。MyBatis-Flex的QueryWrapper覆盖面更广大多数场景都能在Java代码里完成。QueryWrapper的设计差异两个框架的QueryWrapper虽然名字一样但设计思路完全不同。MyBatis-Plus的QueryWrapper是泛型的QueryWrapperT。条件构建有两种方式字符串字段名wrapper.eq(user_name, sam)和Lambda方法引用wrapper.eq(User::getUserName, sam)。字符串方式容易写错字段名Lambda方式解决了这个问题但要求实体类必须有对应的getter方法。MyBatis-Flex的QueryWrapper不带泛型。条件通过APT生成的QueryColumn来构建ORDER.USER_NAME.eq(sam)这种写法。字段引用是编译期常量天然类型安全不依赖实体类的getter方法。还有一个容易忽略的差异MyBatis-Flex的QueryWrapper支持序列化和RPC传输。在微服务架构下一个服务构建的QueryWrapper可以通过RPC传给另一个服务执行。MyBatis-Plus的Wrapper内部持有Lambda表达式引用不支持序列化传输。另外MyBatis-Flex的QueryWrapper在遇到null值时会自动忽略该条件不需要手动判断。MyBatis-Plus需要用wrapper.eq(value ! null, column, value)来处理动态条件。部分字段更新更新订单状态只改status字段其他字段不动。MyBatis-Plus用UpdateWrapper的set方法显式指定要更新的字段UpdateWrapperOrderwrappernewUpdateWrapper();wrapper.eq(id,orderId).set(status,2).set(total_amount,newBigDecimal(0.00));orderMapper.update(null,wrapper);MyBatis-Flex用UpdateEntity只更新调了setter的字段OrderorderUpdateEntity.of(Order.class,orderId);order.setStatus(2);order.setTotalAmount(newBigDecimal(0.00));orderMapper.update(order);UpdateEntity创建的代理对象会记录每个setter调用最终只把这些字段写进UPDATE语句。没调setter的字段不管实体对象里的值是什么都不会出现在SQL里。这个设计还有一个很实用的好处可以把某个字段从有值更新为null。在MyBatis-Plus里updateById默认忽略null值你传个null进去它不更新。想置空某个字段就得额外再调一次updateorderMapper.updateById(order);// 想置空remark还得再补一次if(order.getRemark()null){orderMapper.update(null,newLambdaUpdateWrapperOrder().eq(Order::getId,order.getId()).set(Order::getRemark,null));}这种写法用过的应该都懂不优雅但没办法。UpdateEntity就不存在这个问题你调了order.setRemark(null)它就给你更新为null不需要二次操作。Db Row无实体类操作这是MyBatis-Flex独有的能力MyBatis-Plus没有。Db是一个工具类Row是HashMap的子类。两者配合可以在没有实体类的情况下直接操作数据库RowrownewRow();row.set(order_no,ORD20250703002);row.set(user_id,1004L);row.set(total_amount,newBigDecimal(66.00));row.set(status,0);Db.insert(order,row);适合写临时脚本、做数据迁移、或者处理一些不固定的动态表结构。不需要为每张表都定义一个实体类。查询也行用QueryWrapper构建条件调Db.paginate()就完事了。功能对比整理了一张对比表方便选型时参考对比维度MyBatis-Plus 3.xMyBatis-FlexSQL生成方式启动期注入MappedStatement运行时Provider注解拦截器分页、租户等靠拦截器实现没有拦截器SQL解析拦截器内解析原始SQL不解析直接拼SQL第三方依赖coreextensionstarter只依赖MyBatis条件查询类型安全LambdaQueryWrapper方法引用APT生成QueryColumn编译期常量条件为null时需要手动判断自动忽略分页实现拦截器需额外配置内建在core多表查询需要手写XMLQueryWrapper直接joinQueryWrapper序列化不支持支持RPC传输无实体类操作不支持Db Row部分字段更新UpdateWrapper.set()UpdateEntity.of()多主键/复合主键不支持支持数据脱敏/字段加密收费功能免费生态和社区成熟文档丰富用户多较新社区较小学习成本低上手快需要理解APT小结两个框架不是谁替代谁的关系设计取向不同。MyBatis-Plus走的是「在MyBatis之上尽可能多扩展」的路线。功能全拦截器机制让它的扩展点很多生态也成熟。文档多遇到问题搜一下基本都能找到答案。代价是体积不小拦截器和SQL解析带来额外的复杂度多表查询最终还是要回到XML。MyBatis-Flex走的是「极简轻量」的路线。没有拦截器、没有SQL解析、零第三方依赖QueryWrapper直接支持多表joinAPT生成类型安全的查询条件。这些设计在工程上确实干净。代价是生态薄社区小遇到问题能查的资料不多。老项目用着MyBatis-Plus没必要换生态成熟这个优势不是技术层面能衡量的。新项目如果团队愿意花点时间熟悉MyBatis-Flex值得试试尤其是多表查询多的场景能少写不少XML。参考的内容MyBatis-Flex官网MyBatis-Flex和同类框架功能对比MyBatis-Flex源码版本1.11.8MyBatis-Plus源码版本3.x

相关新闻

xbatis-ddl-auto:轻量自动建表工具,功能丰富且安全有保障!
2026/7/3 22:00:44

xbatis-ddl-auto:轻量自动建表工具,功能丰富且安全有保障!

xbatis-ddl-auto简介xbatis-ddl-auto是一个基于xbatis实体元数据的轻量自动建表工具。它复用xbatis的 Table、TableId、TableField、ColumnDefinition等注解解析结果,根据实体类生成并执行数据库DDL,提供接近JPA ddl-autocreate/update的使用体验&#x…

阅读更多
NoFences:终极免费Windows桌面分区工具,3分钟告别杂乱桌面
2026/7/3 22:00:44

NoFences:终极免费Windows桌面分区工具,3分钟告别杂乱桌面

NoFences:终极免费Windows桌面分区工具,3分钟告别杂乱桌面 【免费下载链接】NoFences 🚧 Open Source Stardock Fences alternative 项目地址: https://gitcode.com/gh_mirrors/no/NoFences 你是否厌倦了在混乱的Windows桌面上寻找文件…

阅读更多
【小白也能轻松玩转龙虾】虾壳云一键部署新手专属包,专门适配零基础用户安装(附最新安装包)
2026/7/3 22:00:44

【小白也能轻松玩转龙虾】虾壳云一键部署新手专属包,专门适配零基础用户安装(附最新安装包)

OpenClaw(小龙虾)Windows 一键部署实操手册|十分钟搭建专属本地数字员工 适配平台:Windows 10/11(64 位)|零基础友好|全可视化界面|无编程门槛 当下热度较高的开源 AI 智…

阅读更多
Umi-OCR 文字识别软件:从零开始掌握免费离线OCR工具
2026/7/3 23:00:44

Umi-OCR 文字识别软件:从零开始掌握免费离线OCR工具

Umi-OCR 文字识别软件:从零开始掌握免费离线OCR工具 【免费下载链接】Umi-OCR OCR software, free and offline. 开源、免费的离线OCR软件。支持截屏/批量导入图片,PDF文档识别,排除水印/页眉页脚,扫描/生成二维码。内置多国语言库…

阅读更多
Midscene.js:基于AI视觉的零代码自动化测试与RPA实践指南
2026/7/3 23:00:44

Midscene.js:基于AI视觉的零代码自动化测试与RPA实践指南

1. 项目概述:当AI视觉遇见零代码自动化最近在折腾自动化测试和RPA(机器人流程自动化)时,我发现了一个挺有意思的开源项目——Midscene.js。简单来说,它想解决一个自动化领域的老大难问题:我们写的那些基于D…

阅读更多
Unitree Go2 ROS2 SDK开发实战:如何为四足机器人构建智能导航系统?
2026/7/3 23:00:44

Unitree Go2 ROS2 SDK开发实战:如何为四足机器人构建智能导航系统?

Unitree Go2 ROS2 SDK开发实战:如何为四足机器人构建智能导航系统? 【免费下载链接】go2_ros2_sdk Unofficial ROS2 SDK support for Unitree GO2 AIR/PRO/EDU 项目地址: https://gitcode.com/gh_mirrors/go/go2_ros2_sdk 想要让你的Unitree Go2四…

阅读更多
STM32L031K6与MC74HC165A的GPIO扩展方案详解
2026/7/3 23:00:44

STM32L031K6与MC74HC165A的GPIO扩展方案详解

1. 为什么需要MC74HC165A与STM32L031K6的组合?在工业控制和嵌入式系统中,我们经常遇到一个经典矛盾:主控芯片的GPIO引脚数量有限,但外部设备需要监测或控制的信号却越来越多。传统解决方案要么使用更昂贵的多引脚MCU,要…

阅读更多
qmcdump:终极QQ音乐解密工具,3步解锁你的加密音乐
2026/7/3 23:00:44

qmcdump:终极QQ音乐解密工具,3步解锁你的加密音乐

qmcdump:终极QQ音乐解密工具,3步解锁你的加密音乐 【免费下载链接】qmcdump 一个简单的QQ音乐解码(qmcflac/qmc0/qmc3 转 flac/mp3),仅为个人学习参考用。 项目地址: https://gitcode.com/gh_mirrors/qm/qmcdump …

阅读更多
程序员的技术水平突飞猛进-最快的方法是什么?
2026/7/3 22:00:44

程序员的技术水平突飞猛进-最快的方法是什么?

我觉得最快的办法,是深度参与1到2个流量非常大的系统。这不是唯一的办法,但是最快的。当然也不是每个人都能遇到这种机会。 你唯一能做的,是好好准备,争取去到一个比较有可能有这样项目的公司和部门。 我自己就是这么过来的。职业…

阅读更多
AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告
2026/7/3 19:49:14

AI Coding 六个月真实ROI账本:产品经理的血泪教训,研发的冷静忠告

6个月前的2025年12月,Boris Cherny 公开宣布自己卸载了 IDE。一时间,Vibe Coding 成了全行业最热的话题。6个月后,当我们回过头来拉一份真实账本,发现事情远没有"一句话生成一个App"那么浪漫。本文从产品经理和研发两个…

阅读更多
审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?
2026/7/3 2:39:23

审计来了,数据权限全开——审计走了,怎么确保权限全部关掉?

引言:审计结束三个月了,审计员的权限还没关某城商行每年按照监管要求开展至少一次数据安全审计。审计期间,内审部门需要抽样检查各类业务数据——交易流水、客户信息、员工操作日志、权限配置记录。这些数据分布在不同系统中,审计…

阅读更多
AutoRaise终极指南:3分钟掌握macOS鼠标悬停自动激活窗口技巧
2026/7/3 0:00:39

AutoRaise终极指南:3分钟掌握macOS鼠标悬停自动激活窗口技巧

AutoRaise终极指南:3分钟掌握macOS鼠标悬停自动激活窗口技巧 【免费下载链接】AutoRaise AutoRaise (and focus) a window when hovering over it with the mouse 项目地址: https://gitcode.com/gh_mirrors/au/AutoRaise AutoRaise是一款革命性的macOS窗口管…

阅读更多
AI Agent五大设计模式解析与实战优化
2026/7/3 0:00:39

AI Agent五大设计模式解析与实战优化

1. AI Agent设计模式全景概览在智能系统开发领域,AI Agent的设计模式就像建筑师的蓝图,决定了智能体如何感知环境、处理信息并采取行动。从业五年来,我见证过太多团队因为模式选择不当导致系统重构的案例。最近在金融风控系统升级时&#xff…

阅读更多
iOS自动化测试:基于facebook-wda与weditor的稳定元素定位实战
2026/7/3 0:00:39

iOS自动化测试:基于facebook-wda与weditor的稳定元素定位实战

1. 项目概述:iOS自动化测试的“定位”之痛做iOS自动化测试的朋友,十有八九都卡在“元素定位”这个环节上。你兴冲冲地写好了测试脚本,结果一运行,要么是找不到元素,要么是找到了但点不动,要么是这次能跑通下…

阅读更多
基于Dify与DeepSeek构建私有知识库问答系统实战指南
2026/7/3 2:40:23

基于Dify与DeepSeek构建私有知识库问答系统实战指南

在业务中快速构建一个能理解私有文档、准确回答专业问题的智能助手,是很多开发团队面临的共同挑战。传统方案往往需要从零开始搭建复杂的 RAG(检索增强生成)系统,涉及文档解析、向量化、检索、大模型调用等多个环节,整…

阅读更多
FAE放射组学分析工具:医学影像特征探索的完整解决方案
2026/7/3 4:59:02

FAE放射组学分析工具:医学影像特征探索的完整解决方案

FAE放射组学分析工具:医学影像特征探索的完整解决方案 【免费下载链接】FAE FeAture Explorer 项目地址: https://gitcode.com/gh_mirrors/fae/FAE 你是否曾经面对海量医学影像数据感到无从下手?想要从CT、MRI等影像中提取有价值的定量特征&#…

阅读更多
DesktopNaotu:你的终极离线思维导图解决方案,告别网络依赖!
2026/7/3 11:08:19

DesktopNaotu:你的终极离线思维导图解决方案,告别网络依赖!

DesktopNaotu:你的终极离线思维导图解决方案,告别网络依赖! 【免费下载链接】DesktopNaotu 桌面版脑图 (百度脑图离线版,思维导图) 跨平台支持 Windows/Linux/Mac OS. (A cross-platform multilingual Mind Map Tool) 项目地址:…

阅读更多