《从 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 吃内存大户!)
  • ​混合架构​ :学会在不同场景选择合适工具,​没有银弹​
相关推荐
海尔辛5 小时前
Unity UI 性能优化--Sprite 篇
ui·unity·性能优化
凌佚8 小时前
rknn优化教程(一)
c++·目标检测·性能优化
yuren_xia8 小时前
Spring Boot + MyBatis 集成支付宝支付流程
spring boot·tomcat·mybatis
橘子青衫9 小时前
Java并发编程利器:CyclicBarrier与CountDownLatch解析
java·后端·性能优化
Elastic 中国社区官方博客10 小时前
Elastic 获得 AWS 教育 ISV 合作伙伴资质,进一步增强教育解决方案产品组合
大数据·人工智能·elasticsearch·搜索引擎·云计算·全文检索·aws
Tom Boom11 小时前
Git常用命令完全指南:从入门到精通
大数据·git·elasticsearch·docker·自动化测试框架
星星点点洲13 小时前
【Elasticsearch】 查询优化方式
elasticsearch·搜索引擎
神仙别闹14 小时前
基于Java(SpringBoot、Mybatis、SpringMvc)+MySQL实现(Web)小二结账系统
java·spring boot·mybatis
@BreCaspian15 小时前
Git 推送失败解决教程——error: failed to push some refs to
大数据·git·elasticsearch
聪颖不聪颖15 小时前
使用 Time Profiler 查看关键函数调用耗时情况,从而分析和解决问题
性能优化