MyBatis-Plus 不只是简化CRUD!资深架构师总结的15个高阶用法

大家好,我是大华!相信Java后端开发的朋友们对MyBatis-Plus(简称MP)肯定不陌生,它可以让CRUD的操作瞬间起飞。

今天就给大家分享15个使用技巧。

1. Service 和 Mapper?

IService :内置了海量现成方法,save, update, list, page 等,适用于绝大多数单表 CRUD。 BaseMapper:当需要进行复杂联表查询、或者要写自定义 SQL 时,它就是你的主战场。

java 复制代码
// 简单查询,用 Service 的 lambdaQuery
List<User> users = userService.lambdaQuery()
                .eq(User::getStatus, 1)
                .like(User::getName, "张")
                .list();

// 复杂查询,用 Mapper + XML 或注解!
// 在 UserMapper.java 中
@Select("SELECT u.*, d.dept_name FROM user u LEFT JOIN dept d ON u.dept_id = d.id WHERE u.id = #{userId}")
UserVO selectUserDetail(@Param("userId") Long userId);

别在 Service 里硬塞复杂 SQL,该写 XML 的时候还是不能偷懒。

2. Lambda 表达式

MP 最棒的特性之一就是 Lambda 查询,编译期就能发现字段名错误,告别运行时才发现拼错的尴尬。

java 复制代码
// 【错误姿势】字段名是字符串,容易拼错,编译器不报错
userService.lambdaQuery().eq("naem", "张三"); // 运行才报错,哭死!

// 【正确姿势】使用方法引用,安全又优雅
userService.lambdaQuery().eq(User::getName, "张三"); 
// 编译不通过,立马改正!

3. 分页查询,不只是PageHelper的替代品

MP 的分页功能非常强大,而且与自身条件构造器无缝集成。

java 复制代码
// 创建分页参数,并指定排序
Page<User> page = new Page<>(1, 20); // 查第1页,每页20条
page.addOrder(OrderItem.desc("create_time")); // 按创建时间倒序

// 执行分页查询
Page<User> userPage = userService.page(page,
        Wrappers.<User>lambdaQuery()
                .eq(User::getDeptId, 2)
);

// 直接转换为 VO 分页对象,一步到位
Page<UserVO> voPage = userPage.convert(user -> {
    UserVO vo = new UserVO();
    BeanUtils.copyProperties(user, vo); // 使用 Spring 的工具类
    // 或者用 MapStruct 等更专业的工具
    return vo;
});

4. 批量操作,性能提升的关键

大批量数据插入/更新时,一条条处理会让数据库哭泣。

java 复制代码
// 批量插入,分批提交
List<User> hugeUserList = ... // 一个巨大的列表
userService.saveBatch(hugeUserList, 1000); // 每1000条批量提交一次

// 批量更新(自己控制事务)
@Transactional(rollbackFor = Exception.class)
public void batchUpdateStatus(List<Long> ids, Integer status) {
    List<User> updateList = ids.stream().map(id -> {
        User user = new User();
        user.setId(id);
        user.setStatus(status);
        return user;
    }).collect(Collectors.toList());
    userService.updateBatchById(updateList);
}

5. 条件构造器,让你的逻辑更清晰

QueryWrapperLambdaQueryWrapper可以构建非常复杂的查询逻辑。

java 复制代码
// 复杂的 AND-OR 组合查询
List<User> users = userService.lambdaQuery()
    .eq(User::getStatus, 1)
    .and(wrapper -> wrapper // 这是一个 AND 嵌套
        .like(User::getName, "张")
        .or()
        .like(User::getEmail, "zhang") // name LIKE '%张%' OR email LIKE '%zhang%'
    )
    .between(User::getCreateTime, startTime, endTime)
    .list();

// 【性能技巧】只查需要的字段,避免 SELECT *
List<User> userList = userService.lambdaQuery()
    .select(User::getId, User::getName) // 只查询 ID 和 Name 字段
    .eq(User::getStatus, 1)
    .list();

6. 自动填充,告别手动 set 创建时间

create_time, update_time 这种字段,就别再手动 set 了。

java 复制代码
@Component
public class MyMetaObjectHandler implements MetaObjectHandler {

    @Override
    public void insertFill(MetaObject metaObject) {
        // 插入时自动填充
        this.strictInsertFill(metaObject, "createTime", LocalDateTime.class, LocalDateTime.now());
        this.strictInsertFill(metaObject, "createBy", String.class, getCurrentUser());
    }

    @Override
    public void updateFill(MetaObject metaObject) {
        // 更新时自动填充
        this.strictUpdateFill(metaObject, "updateTime", LocalDateTime.class, LocalDateTime.now());
    }
}

// 实体类字段上需要加注解
public class User {
    @TableField(fill = FieldFill.INSERT)
    private LocalDateTime createTime;
    
    @TableField(fill = FieldFill.INSERT_UPDATE)
    private LocalDateTime updateTime;
}

7. 逻辑删除,数据不是真的删除

千万别用 delete from 硬删数据了!

yaml 复制代码
# 在 application.yml 中配置
mybatis-plus:
  global-config:
    db-config:
      logic-delete-field: deleted  # 全局逻辑删除实体字段名
      logic-delete-value: 1        # 逻辑已删除值(默认为 1)
      logic-not-delete-value: 0    # 逻辑未删除值(默认为 0)

配置后,调用 userService.removeById(1),MP 实际执行的是: UPDATE user SET deleted = 1 WHERE id = 1 AND deleted = 0。 所有查询也会自动带上 AND deleted = 0 条件。

8. 枚举处理器,告别数据库存数字的迷惑行为

数据库存 status = 1,代码里还要猜 1 是啥意思?用枚举!

java 复制代码
@Getter
public enum UserStatus {
    ENABLED(1, "启用"),
    DISABLED(0, "禁用");

    private final int code;
    private final String desc;

    UserStatus(int code, String desc) {
        this.code = code;
        this.desc = desc;
    }
}

// 实体类中直接使用枚举类型
public class User {
    private UserStatus status;
}

// 配置枚举处理器(Spring Boot 基本不用配了,开箱即用)

这样,数据库存的是数字 1,但代码里操作的一直是 UserStatus.ENABLED,清晰明了!

9. 多租户数据隔离,SAAS 系统必备

SAAS 应用中,不同租户的数据必须严格隔离。MP 的租户插件可以自动在每次查询时加上租户 ID。

java 复制代码
@Configuration
public class MybatisPlusConfig {

    @Bean
    public MybatisPlusInterceptor mybatisPlusInterceptor() {
        MybatisPlusInterceptor interceptor = new MybatisPlusInterceptor();
        
        // 租户插件
        TenantLineInnerInterceptor tenantInterceptor = new TenantLineInnerInterceptor();
        tenantInterceptor.setTenantLineHandler(new TenantLineHandler() {
            @Override
            public Expression getTenantId() {
                // 从当前上下文中获取租户ID,比如从 JWT Token 中
                return new StringValue(TenantContext.getCurrentTenantId());
            }
            
            @Override
            public String getTenantIdColumn() {
                return "tenant_id"; // 数据库中的租户ID列名
            }
            
            @Override
            public boolean ignoreTable(String tableName) {
                // 忽略不需要租户隔离的表,如全局配置表
                return "system_config".equals(tableName);
            }
        });
        interceptor.addInnerInterceptor(tenantInterceptor);
        return interceptor;
    }
}

10. 代码生成器,效率翻倍利器

别再手撸 Entity, Mapper, Service, Controller 了!

java 复制代码
public class CodeGenerator {
    public static void main(String[] args) {
        AutoGenerator generator = new AutoGenerator();
        
        // 数据源配置
        DataSourceConfig dataSourceConfig = new DataSourceConfig();
        dataSourceConfig.setUrl("jdbc:mysql://localhost:3306/test");
        dataSourceConfig.setDriverName("com.mysql.cj.jdbc.Driver");
        dataSourceConfig.setUsername("root");
        dataSourceConfig.setPassword("123456");
        generator.setDataSource(dataSourceConfig);
        
        // 全局配置
        GlobalConfig globalConfig = new GlobalConfig();
        globalConfig.setOutputDir(System.getProperty("user.dir") + "/src/main/java");
        globalConfig.setAuthor("大华");
        globalConfig.setOpen(false);
        globalConfig.setSwagger2(true); // 实体属性 Swagger2 注解
        generator.setGlobalConfig(globalConfig);
        
        // 包配置
        PackageConfig packageConfig = new PackageConfig();
        packageConfig.setParent("com.laomao.demo");
        packageConfig.setEntity("domain.entity");
        packageConfig.setMapper("dao.mapper");
        packageConfig.setService("service");
        packageConfig.setServiceImpl("service.impl");
        generator.setPackageInfo(packageConfig);
        
        generator.execute(); // 执行生成
    }
}

运行一下,全套代码瞬间生成!

11. 自定义全局拦截器,统一处理逻辑

可以用来做数据权限控制、SQL 性能监控、字段加解密等。

java 复制代码
@Component
@Slf4j
public class SqlLogInterceptor implements InnerInterceptor {
    
    @Override
    public void beforeQuery(Executor executor, MappedStatement ms, Object parameter, BoundSql boundSql) {
        long start = System.currentTimeMillis();
        // 将开始时间存入当前线程上下文
    }
    
    @Override
    public void afterQuery(Executor executor, MappedStatement ms, Object parameter, BoundSql boundSql, List<Object> result) {
        long end = System.currentTimeMillis();
        long cost = end - start;
        if (cost > 1000) { // 超过1秒算慢SQL
            log.warn("慢SQL警告: {}, 执行耗时: {}ms", boundSql.getSql(), cost);
            // 可以接入告警系统,通知开发人员
        }
    }
}

12. 分布式主键 ID,告别数据库自增

在分布式系统中,数据库自增 ID 是瓶颈。推荐使用雪花算法。

java 复制代码
public class User {
    // 指定主键类型为 ASSIGN_ID(雪花算法)
    @TableId(type = IdType.ASSIGN_ID)
    private Long id; // 注意是 Long,不是 Integer
    // ...
}

13. 乐观锁,防止并发更新覆盖

高并发下,防止后提交的数据覆盖先提交的数据。

java 复制代码
// 实体类中增加版本号字段
public class User {
    @Version
    private Integer version;
}

// 更新时,MP会自动带上版本号条件
User user = userService.getById(1L);
user.setName("新名字");
userService.updateById(user); // SQL: UPDATE user SET name=?, version=? WHERE id=? AND version=?

14. 结果映射,自动处理一对一、一对多

MP 可以和 MyBatis 的 @Result 注解完美结合。

java 复制代码
// 在 Mapper 方法上使用复杂结果映射
@Select("SELECT u.*, d.name as dept_name FROM user u LEFT JOIN department d ON u.dept_id = d.id WHERE u.id = #{id}")
@Results({
    @Result(column = "id", property = "id"),
    @Result(column = "dept_name", property = "deptName"),
    @Result(column = "id", property = "roles", 
            many = @Many(select = "com.laomao.mapper.RoleMapper.findByUserId"))
})
UserVO findUserWithDept(Long id);

15. 事务管理,保证数据一致性

这是最后一道防线,也是最关键的一道。

java 复制代码
@Service
public class UserService {
    
    @Transactional(rollbackFor = Exception.class) // 注意:默认只回滚 RuntimeException
    public void createUserWithInitData(User user) {
        // 1. 保存用户基本信息
        userService.save(user);
        
        // 2. 初始化用户账户
        Account account = new Account();
        account.setUserId(user.getId());
        accountService.save(account);
        
        // 3. 发送欢迎消息(如果消息发送失败,希望用户创建也回滚)
        messageService.sendWelcomeMessage(user.getId());
        
        // 任何一个步骤出错,所有操作都会回滚
    }
}

总结

MyBatis-Plus 的强大远不止于此,但掌握以上 15个核心技巧,足以让你在日常开发中游刃有余,写出既高效又优雅的代码。

本文首发于公众号:程序员刘大华,专注分享前后端开发的实战笔记。关注我,少走弯路,一起进步!

📌往期精彩

《这20条SQL优化方案,让你的数据库查询速度提升10倍》

《MySQL 为什么不推荐用雪花ID 和 UUID 做主键?》

《图片标签用 img 还是 picture?很多人彻底弄混了》

《还在用 WebSocket 做实时通信?SSE 可能更简单》

相关推荐
吴名氏.2 小时前
电子书《ASP.NET MVC企业级实战》
后端·asp.net·mvc·编程语言
IT_陈寒2 小时前
SpringBoot 3.2新特性实战:这5个隐藏技巧让你的应用性能飙升50%
前端·人工智能·后端
Han.miracle2 小时前
Java EE --JUC
java·线程池·原子类·callable·semaphore·reentrantlcok
那我掉的头发算什么2 小时前
【javaEE】多线程——线程安全初阶☆☆☆
java·jvm·安全·java-ee·intellij-idea
yzp-2 小时前
Zookeeper 笔记
java·分布式·zookeeper
蜡笔大新7982 小时前
IDEA中的异常
java·ide·intellij-idea
小奏技术3 小时前
LLM 交互的“省钱”新姿势:JSON 已死,TOON 当立
后端·aigc
用户21411832636023 小时前
mcp-server案例分享-即梦MCP-Server实战教程-让Claude直接调用AI生图视频能力
后端