【SpringBoot+Elasticsearch 内容搜索系统实战】:架构设计与全流程实现

🔥你好我是fengxin_rou这是我的个人主页 fengxin_rou的主页

❄️欢迎查看我的专栏我的专栏

《Java后端学习》《JAVASE基础》《JUC并发》《redis》《JVM虚拟机》《MYSQL》《黑马点评》《rabbitmq》《JavaWeb+AI的talis学习系统》《苍穹外卖》

目录

前言

[一、Elasticsearch 索引设计与初始化](#一、Elasticsearch 索引设计与初始化)

[1.1 核心概念类比](#1.1 核心概念类比)

[1.2 索引初始化实现](#1.2 索引初始化实现)

[1.3 字段设计要点](#1.3 字段设计要点)

二、搜索索引数据写入与同步机制

[2.1 全量数据回灌](#2.1 全量数据回灌)

[2.2 单篇文档写入逻辑](#2.2 单篇文档写入逻辑)

[2.3 软删除实现](#2.3 软删除实现)

[三、基于 Kafka+Canal 的增量数据同步](#三、基于 Kafka+Canal 的增量数据同步)

[3.1 同步架构](#3.1 同步架构)

[3.2 消息消费逻辑](#3.2 消息消费逻辑)

[3.3 优势说明](#3.3 优势说明)

四、搜索服务核心实现:检索、加权与分页

[4.1 完整搜索流程](#4.1 完整搜索流程)

[4.2 多字段匹配与权重加权](#4.2 多字段匹配与权重加权)

[4.3 游标分页实现](#4.3 游标分页实现)

[4.4 高亮与摘要生成](#4.4 高亮与摘要生成)

五、搜索建议功能实现

结语

前言

在内容平台场景中,高性能、高相关性、实时可搜是搜索模块的核心诉求。本文基于 SpringBoot 与 Elasticsearch(ES),从零实现一套包含索引初始化、数据同步、增量更新、关键词检索、游标分页的完整搜索服务,解决传统数据库搜索性能差、分词不精准、实时性不足等痛点,可直接应用于文章、资讯、社区类内容平台。

一、Elasticsearch 索引设计与初始化

1.1 核心概念类比

ES 是分布式搜索引擎,核心是倒排索引,其结构可与 MySQL 直接类比,降低理解成本:

  • Index(索引)≈ 数据库表
  • Document(文档)≈ 表行数据
  • Mapping(映射)≈ 表结构 Schema
  • Field(字段)≈ 表列

1.2 索引初始化实现

项目启动时自动创建索引与 Mapping,title/body 字段启用 IK 分词,需提前安装 ES 分析 - ik 插件。标题使用 ik_max_word 分词、ik_smart 检索,兼顾召回率与精准度。

复制代码
/**
 * 搜索索引初始化:应用启动时创建索引与映射
 */
@Service
@RequiredArgsConstructor
public class SearchIndexInitializer {
    private final ElasticsearchClient es;
    private static final String INDEX = "zhiguang_content_index";

    @PostConstruct
    public void ensureIndex() {
        try {
            // 检查索引是否存在
            boolean exists = es.indices().exists(e -> e.index(INDEX)).value();
            if (exists) return;
            // 创建索引并定义映射
            es.indices().create(c -> c.index(INDEX).mappings(m -> m
                .properties("content_id", p -> p.long_(LongNumberProperty.of(b -> b)))
                .properties("title", p -> p.text(t -> t.analyzer("ik_max_word").searchAnalyzer("ik_smart")))
                .properties("body", p -> p.text(t -> t.analyzer("ik_max_word")))
                .properties("status", p -> p.keyword(KeywordProperty.of(b -> b)))
                .properties("title_suggest", p -> p.completion(CompletionProperty.of(b -> b)))
                // 其他字段省略...
            ));
        } catch (Exception ignored) {}
    }
}

1.3 字段设计要点

  • keyword 类型 :用于标签、状态、作者信息等精确匹配与过滤,不分词。
  • text 类型 :用于标题、正文等全文检索,绑定 IK 分词器。
  • completion 类型 :专门用于搜索建议,提升输入联想体验。

二、搜索索引数据写入与同步机制

2.1 全量数据回灌

应用启动时若索引为空,自动从数据库分页读取历史数据,批量写入 ES,保证索引数据完整。

复制代码
@PostConstruct
public void ensureBackfill() {
    long cnt = es.count(c -> c.index(INDEX)).count();
    if (cnt > 0) return;
    int limit = 500;
    int offset = 0;
    while (true) {
        List<KnowPostFeedRow> rows = knowPostMapper.listFeedPublic(limit, offset);
        if (rows == null || rows.isEmpty()) break;
        for (KnowPostFeedRow r : rows) {
            upsertKnowPost(r.getId());
        }
        offset += rows.size();
    }
}

2.2 单篇文档写入逻辑

核心方法 upsertKnowPost 实现数据新增 / 更新,流程标准化:

  1. 从数据库查询文章详情;
  2. 远程拉取正文,失败则使用描述兜底,截断至 4000 字符;
  3. 补充点赞、收藏等计数数据;
  4. 写入 ES 并设置 refresh=WaitFor保证写入后立即可搜

2.3 软删除实现

不物理删除文档,仅更新 status=deleted,搜索时过滤该状态,避免数据丢失与索引波动。

复制代码
public void softDeleteKnowPost(long id) {
    Map<String, Object> doc = new HashMap<>();
    doc.put("content_id", id);
    doc.put("status", "deleted");
    es.index(i -> i.index(INDEX).id(String.valueOf(id))
        .document(doc).refresh(Refresh.WaitFor));
}

三、基于 Kafka+Canal 的增量数据同步

3.1 同步架构

使用 Canal 监听 MySQL binlog ,将数据变更发送至 Kafka 的 canal-outbox 主题,搜索模块作为消费者,实现数据库与 ES 数据准实时一致

3.2 消息消费逻辑

与用户关系模块共用 Topic,通过不同消费者组 隔离业务,仅处理 entity=knowpost 的变更消息,保证幂等性。

复制代码
/**
 * 搜索索引 Outbox 消费者
 */
@Service
@RequiredArgsConstructor
public class CanalOutboxConsumerSearch {
    private final SearchIndexService indexService;

    @KafkaListener(topics = OutboxTopics.CANAL_OUTBOX, groupId = "search-index-consumer")
    public void onMessage(String message, Acknowledgment ack) {
        try {
            List<JsonNode> rows = OutboxMessageUtil.extractRows(objectMapper, message);
            for (JsonNode row : rows) {
                JsonNode payload = objectMapper.readTree(row.get("payload").asText());
                String entity = payload.get("entity").asText();
                String op = payload.get("op").asText();
                Long id = payload.get("id").asLong();
                if (!"knowpost".equals(entity) || id == null) continue;
                // 执行更新或软删除
                if ("delete".equalsIgnoreCase(op)) {
                    indexService.softDeleteKnowPost(id);
                } else {
                    indexService.upsertKnowPost(id);
                }
            }
            ack.acknowledge();
        } catch (Exception ignored) {}
    }
}

3.3 优势说明

  • 解耦:数据库变更与搜索同步分离,互不影响;
  • 高可用:消息队列缓冲流量,避免直接写入 ES 导致雪崩;
  • 易扩展:新增下游模块只需新增消费者组,无侵入改造。

四、搜索服务核心实现:检索、加权与分页

4.1 完整搜索流程

前端传入关键词、标签、分页参数,后端构建 ES 查询,流程分为:参数解析→召回过滤→业务加权→排序高亮→游标分页→结果封装

4.2 多字段匹配与权重加权

使用 multi_match 实现多字段检索,标题权重设为 3 ,正文权重为 1,提升标题匹配优先级。通过 function_score 对点赞、浏览量做对数加权,让优质内容排名更靠前。

复制代码
// 构建查询核心逻辑
.query(qb -> qb.functionScore(fs -> fs
    .query(qb2 -> qb2.bool(bq -> {
        // 多字段匹配,标题权重3倍
        bq.must(m -> m.multiMatch(mm -> mm.query(q).fields("title^3", "body")));
        // 过滤已发布内容
        bq.filter(f -> f.term(t -> t.field("status").value("published")));
        // 标签过滤
        if (!tags.isEmpty()) {
            bq.filter(f -> f.terms(t -> t.field("tags").terms(tv -> tv.value(tags))));
        }
        return bq;
    }))
    // 点赞数加权:log(1+like)×2
    .functions(fn -> fn.fieldValueFactor(f -> f.field("like_count").modifier(Log1p)).weight(2.0))
    // 浏览数加权:log(1+view)×1
    .functions(fn -> fn.fieldValueFactor(f -> f.field("view_count").modifier(Log1p)).weight(1.0))
    .boostMode(Sum)
))

4.3 游标分页实现

替代传统 offset+limit,使用 search_after 实现深分页高性能,将最后一条数据的排序值(评分、时间、点赞、ID)Base64 编码为游标,下一页从该位置继续查询。

4.4 高亮与摘要生成

对标题、正文关键词添加 <em> 高亮标签,合并为搜索摘要(Snippet),提升用户阅读体验。

五、搜索建议功能实现

基于 ES completion 类型实现输入联想,用户输入前缀时快速返回标题候选,响应时间毫秒级。

复制代码
public SuggestResponse suggest(String prefix, int size) {
    var resp = es.search(s -> s.index(INDEX)
        .suggest(sug -> sug.suggesters("title_suggest",
            sc -> sc.prefix(prefix).completion(c -> c.field("title_suggest").size(size))))
        , Map.class);
    // 解析建议结果并返回
    List<String> items = new ArrayList<>();
    resp.suggest().get("title_suggest").forEach(s -> {
        s.completion().options().forEach(opt -> items.add(opt.text()));
    });
    return new SuggestResponse(items);
}

结语

本文完整实现了 SpringBoot 整合 Elasticsearch 的企业级内容搜索系统 ,覆盖索引设计、数据全量 / 增量同步、关键词检索、游标分页、搜索建议全流程。方案具备实时性高、检索精准、扩展性强、性能稳定等特点,适配文章、社区、电商等内容搜索场景。

实际落地需注意:IK 分词器自定义词库优化、ES 集群分片规划、异步同步重试机制、查询性能监控。后续可扩展语义搜索、个性化排序、搜索热词统计等能力,进一步提升搜索体验。

相关推荐
逸Y 仙X14 小时前
文章三:Elasticsearch 集群恢复和索引分布
java·大数据·linux·服务器·elasticsearch·搜索引擎·全文检索
还是鼠鼠14 小时前
AI掘金头条新闻系统 (Toutiao News)-用户注册-生成Token
后端·python·mysql·fastapi·web
自珍JAVA20 小时前
访问者模式:让你的代码优雅地“拜访”对象结构
后端
毅航1 天前
AI 浪潮下,会用工具不等于具备能力
后端·程序员·ai编程
比特森林探险记1 天前
go 语言中的context 解读和用法
开发语言·后端·golang
刀法如飞1 天前
《道德经》简单解说版-第 2 章:天下皆知美之为美
前端·后端·面试
绝知此事1 天前
Netty实战:从零构建高性能TCP通信服务(含心跳检测)
java·网络·spring boot·网络协议·tcp/ip
IT_陈寒1 天前
Vue的computed属性怎么突然不更新了?
前端·人工智能·后端
invicinble1 天前
spring提供的其他机制
java·后端·spring