《从 MyBatis-Plus 到 Elasticsearch:一个后端的性能优化踩坑实录》

最近接手了一个老项目,单表查询用 MyBatis-Plus 写得飞起,但一到​​多表关联+模糊搜索​ ​就卡成 PPT。痛定思痛,决定引入 Elasticsearch 优化查询性能,结果踩坑无数......记录下这次​​从 ORM 到搜索引擎​​的升级历程,分享给同样被慢查询折磨的你。

1. 问题定位:MP 的舒适区边界​

MyBatis-Plus 的 QueryWrapper 在单表操作中确实优雅:

复制代码
// 条件查询示例(单表)
List<User> users = userService.lambdaQuery()
    .eq(User::getStatus, 1)
    .like(User::getName, "张")
    .list();

但遇到​​跨表 JOIN + 高并发模糊查询​​时,MySQL 直接裂开:

  • 大表 LIKE '%关键词%' 导致全表扫描
  • 分页 COUNT 计算拖慢响应(即使用了 PageHelper

​结论​​:MP 适合 OLTP 简单场景,OLAP 复杂查询得换方案。


​2. 技术选型:ES 还是 ClickHouse?​

面对海量数据搜索,纠结了两个方案:

方案 优点 缺点
​Elasticsearch​ 实时搜索快,支持分词高亮 资源占用高,学习曲线陡峭
​ClickHouse​ 列式存储,聚合分析强 不适合频繁更新/点查

最终选择 ​​ES​​,因为:

  1. 业务需求以​模糊搜索+排序​为主(如商品名称、用户昵称)
  2. 已有 ES 集群,可复用运维资源

​3. 实战:SpringBoot 集成 ES​

​(1) 引入依赖​
复制代码
<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
</dependency>

注意版本匹配(Spring Data ES 和 ES 服务端版本需兼容)。

​(2) 实体类映射​

用注解定义 ES 索引结构和分词器:

复制代码
@Document(indexName = "products", createIndex = false)
public class Product {
    @Id
    private Long id;
    
    @Field(type = FieldType.Text, analyzer = "ik_max_word") // 中文分词
    private String name;
    
    @Field(type = FieldType.Double)
    private Double price;
}
​(3) 混合查询策略​

​写操作​​:双写 MySQL 和 ES(本地事务+消息队列补偿)

复制代码
@Transactional
public void addProduct(Product product) {
    // 1. 写入 MySQL
    productMapper.insert(product); 
    // 2. 发MQ消息异步写入ES(防止事务失败污染ES)
    rocketMQTemplate.send("es-update-topic", product);
}

​读操作​​:走 ES 查询,兜底查 MySQL

复制代码
public Page<Product> search(String keyword, int page, int size) {
    NativeSearchQuery query = new NativeSearchQueryBuilder()
        .withQuery(QueryBuilders.matchQuery("name", keyword)) // 分词匹配
        .withPageable(PageRequest.of(page, size))
        .build();
    return elasticsearchRestTemplate.search(query, Product.class);
}

​4. 性能对比​

压测结果(100万数据,并发 200):

查询类型 MySQL + MP ES
单字段精确查询 120ms 15ms(↑8倍)
多字段模糊查询 2800ms(全表扫) 50ms(↑56倍)
排序分页 900ms 30ms(↑30倍)

代价:ES 索引延迟约 1s(近实时),存储占用是 MySQL 的 2 倍。


​5. 深度踩坑​

  1. ​分词器选型​ :默认分词器对中文不友好,需安装 ik 插件
  2. ​数据同步一致性​ :双写时用 canal 监听 MySQL Binlog 更可靠
  3. ​ES 动态映射陷阱​:字段类型自动推断可能导致查询异常,建议预定义 mapping

​总结​​:

  • ​简单 CRUD​:继续用 MyBatis-Plus,别过度设计
  • ​复杂搜索/聚合​ :ES 是真香,但要做好​资源隔离+监控​(ES 吃内存大户!)
  • ​混合架构​ :学会在不同场景选择合适工具,​没有银弹​
相关推荐
python算法(魔法师版)8 小时前
数据库故障排查指南:从连接问题和性能优化
服务器·网络·数据库·性能优化
程序员的世界你不懂9 小时前
tomcat6性能优化
前端·性能优化·firefox
破烂公司一级特派员9 小时前
前端开发实战:用React Hooks优化你的组件性能
性能优化·实战·react·前端开发·hooks
A-花开堪折10 小时前
RK3568-OpenHarmony(1) : OpenHarmony 5.1的编译
大数据·elasticsearch·搜索引擎
GISer_Jing10 小时前
前端性能优化全攻略:从基础体验到首屏加载的深度实践
前端·javascript·性能优化
Kx…………12 小时前
Java EE(Spring+Spring MVC+MyBatis)从入门到精通企业级应用开发教程——1初识MyBatis框架
学习·spring·java-ee·mvc·mybatis
码农飞哥12 小时前
互联网大厂Java求职面试实战:Spring Boot微服务与数据库优化详解
java·spring boot·微服务·mybatis·数据库优化·性能监控·安全框架
python算法(魔法师版)13 小时前
JavaScript性能优化实战,从理论到落地的全面指南
开发语言·性能优化·前端框架·代理模式
前端小巷子17 小时前
CSS渲染性能优化
前端·css·面试·性能优化
斯普信专业组17 小时前
Elasticsearch内存管理与JVM优化:原理剖析与最佳实践
大数据·jvm·elasticsearch