dbVisitor 使用 MyBatis 方式操作 ElasticSearch

在 Java 生态中,操作 ElasticSearch 最常见的方式莫过于使用官方的 elasticsearch-java (或旧版的 RestHighLevelClient) 或者 Spring 家族的 spring-data-elasticsearch。这些工具非常强大,但对于习惯了关系型数据库(RDBMS)和 MyBatis 开发模式的开发者来说,切换到 ElasticSearch 往往意味着需要适应一套全新的 API 和思维模式(DSL 构建、Builder 模式等)。

特别是在一个混合架构的项目中,如果同时存在 MySQL 和 ElasticSearch,数据访问层的代码风格割裂感会非常强:一边是 MyBatis 的 Mapper 接口和 XML,另一边是复杂的 DSL 构建代码或 Repository 接口。这种差异不仅增加了学习成本,也让诸如"分页查询"这样的通用功能难以统一实现。

本文将介绍如何使用 dbVisitor,以一种"类 MyBatis"的方式来操作 ElasticSearch,实现架构上的统一。

1. 传统方式的痛点

在传统的混合架构中,我们可能会遇到以下问题:

  • API 风格不统一:RDBMS 使用 SQL 和 JDBC,ElasticSearch 使用 REST API 和 JSON DSL。
  • 分页实现差异 :MyBatis 通常配合 PageHelper 或 RowBounds,而 ElasticSearch 需要手动设置 fromsize,或者使用 Spring Data 的 Pageable
  • 维护成本高:需要维护两套完全不同的底层逻辑,增加了代码的复杂度和出错的概率。

2. dbVisitor 的解决方案

dbVisitor 通过提供一个 JDBC 驱动层(dbvisitor-driver)和适配器(jdbc-elastic),将 ElasticSearch 的操作封装成了标准的 JDBC 接口。这意味着你可以像操作 MySQL 一样操作 ElasticSearch。

更进一步,dbVisitor 提供了类似 MyBatis 的 ORM 功能,支持 Mapper 接口、XML 映射文件、注解以及 Lambda 表达式。

2.1 对象关系映射 (ORM)

首先,我们定义一个 Java 对象,并使用注解进行映射。这与 MyBatis Plus 或 JPA 非常相似。

java 复制代码
@Table("user_info")
public class UserInfo {
    // 映射 _id 字段
    @Column(value = "_id", primary = true)
    private String id;

    @Column("name")
    private String name;

    @Column("age")
    private Integer age;

    // 省略 getter/setter
}

2.2 使用 Mapper 接口 (注解方式)

你可以定义一个 Mapper 接口,使用注解来编写 ElasticSearch 的 DSL 命令。

java 复制代码
@SimpleMapper
public interface UserInfoMapper {
    // 插入数据
    @Insert("POST /user_info/_doc { "name": #{info.name}, "age": #{info.age} }")
    int saveUser(@Param("info") UserInfo info);

    // 根据 ID 查询
    @Query("GET /user_info/_doc/#{id}")
    UserInfo loadById(@Param("id") String id);

    // 删除数据
    @Delete("DELETE /user_info/_doc/#{id}")
    int deleteUser(@Param("id") String id);
}

2.3 使用通用 Mapper

如果你不想写任何命令,可以直接继承 BaseMapper,dbVisitor 会自动生成基础的 CRUD 操作。

java 复制代码
@SimpleMapper
public interface UserInfoBaseMapper extends BaseMapper<UserInfo> {
    // 自动拥有 insert, update, delete, selectById, listBySample 等方法
}

2.4 使用 Lambda 方式

dbVisitor 也提供了类似 MyBatis Plus 的 Lambda 调用方式,完全类型安全。

java 复制代码
LambdaTemplate lambda = new LambdaTemplate(connection);

// 查询 name = "mali" 的用户
UserInfo user = lambda.query(UserInfo.class)
    .eq(UserInfo::getName, "mali")
    .queryForObject();

// 更新操作
lambda.update(UserInfo.class)
    .eq(UserInfo::getId, user.getId())
    .updateTo(UserInfo::getAge, 27)
    .doUpdate();

2.5 使用 XML 管理 Mapper (MyBatis 风格)

对于复杂的查询或需要统一管理 DSL 的场景,dbVisitor 支持使用 XML 文件来定义 Mapper,这与 MyBatis 的体验几乎一致。

Mapper 接口:

java 复制代码
@RefMapper("mapper/user-mapper.xml")
public interface UserInfoXmlMapper {
    int saveUser(@Param("info") UserInfo info);
    List<UserInfo> listByUserName(@Param("userName") String userName, Page page);
}

XML 文件 (user-mapper.xml):

xml 复制代码
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE mapper PUBLIC "-//dbvisitor.net//DTD Mapper 1.0//EN"
        "https://www.dbvisitor.net/schema/dbvisitor-mapper.dtd">
<mapper namespace="com.example.mapper.UserInfoXmlMapper">
    <resultMap id="userResultMap" type="com.example.entity.UserInfo">
        <result column="_id" property="id"/>
        <result column="name" property="name"/>
        <result column="age" property="age"/>
    </resultMap>

    <insert id="saveUser">
        POST /user_info/_doc {
            "name": #{info.name},
            "age": #{info.age}
        }
    </insert>

    <!-- 支持自动分页 -->
    <select id="listByUserName" resultMap="userResultMap">
        POST /user_info/_search {
            "query": {
                "term": { "name": #{userName} }
            }
        }
    </select>
</mapper>

3. 统一的分页实现

在 dbVisitor 中,无论是操作 MySQL 还是 ElasticSearch,分页查询的实现方式是完全统一的。你只需要传递一个 Page 对象。

java 复制代码
// 创建分页对象
Page page = new PageObject();
page.setPageSize(10);
page.setPageNumber(0); // 第一页

// 执行查询,dbVisitor 会自动拦截并重写为分页查询
// 对于 ElasticSearch,会自动转换为 "from": 0, "size": 10
List<UserInfo> list = mapper.listByUserName("mali", page);

// 获取总记录数(如果需要)
long total = page.getTotalCount();

// 翻页
page.nextPage();
list = mapper.listByUserName("mali", page);

4. 同类工具对比

在 Java 生态中,针对 ElasticSearch 的 ORM 封装百花齐放。以下是 dbVisitor 与几款代表性工具的对比,帮助你做出技术选型:

特性 Official Client Spring Data ES Easy-Es dbVisitor
定位 官方底层客户端 Spring 生态标准组件 ES 界的 MyBatis-Plus JDBC 驱动 + ORM
依赖程度 无 (基础库) 强依赖 Spring 依赖 Spring / MP 极低 (仅 JDK + 官方驱动)
API 风格 Builder / DSL Repository / JPA Lambda / Wrapper JDBC / Mapper / XML / Lambda
动态 SQL 需手动拼接 JSON 较弱 不支持 XML 强 (XML 动态标签)
JDBC 支持 原生支持
学习曲线 高 (需熟记 DSL) 中 (需懂 Spring Data) 中 (需懂 MP) 低 (兼容 MyBatis/JDBC 习惯)
统一性 仅限 ES 需借助 Spring 生态 仅限 ES 一套 API 统一操作 RDBMS 和 ES

选型建议

  • Official Client: 如果你需要使用 ES 的最新特性,或者对性能有极致要求,且不介意编写复杂的 DSL 代码。
  • Spring Data Elasticsearch: 如果你是 Spring 全家桶用户,且习惯 JPA/Repository 开发模式。
  • Easy-Es: 如果你是 MyBatis-Plus 的重度用户,希望在 ES 中也能获得类似的开发体验。
  • dbVisitor : 如果你希望用 标准 JDBCMyBatis XML 的方式统一管理 RDBMS 和 ES,或者你的项目不依赖 Spring (如 Solon, Hasor, 纯 Java),dbVisitor 是最佳选择。

5. 总结

通过 dbVisitor,我们可以在同一个项目中,用同一套 API、同一种思维方式(Mapper/XML/Lambda)同时操作关系型数据库和 ElasticSearch。这极大地降低了混合架构项目的开发和维护成本,让数据访问层变得更加整洁和统一。

如果你正在寻找一种能够统一 RDBMS 和 NoSQL 开发体验的工具,dbVisitor 绝对值得一试。

相关推荐
小龙5 小时前
[Git 报错解决]本地分支落后于远程分支(`non-fast-forward`)
大数据·git·elasticsearch·github
DKunYu7 小时前
2.分支管理
大数据·git·elasticsearch·搜索引擎·gitee
Elastic 中国社区官方博客9 小时前
使用 LangGraph 和 Elasticsearch 构建人机交互 Agents
大数据·人工智能·elasticsearch·搜索引擎·langchain·全文检索·人机交互
KANGBboy11 小时前
ES 索引切换及验证
大数据·elasticsearch
DKunYu12 小时前
3.远程操作
大数据·git·elasticsearch·搜索引擎·gitee
禾黍黎14 小时前
ElasticSearch+Logstash 对 数据库数据进行转换和检索
大数据·数据库·elasticsearch
青鱼入云14 小时前
详细介绍下Elasticsearch 布尔查询
大数据·elasticsearch·搜索引擎
神秘代码行者1 天前
Git Restore 命令教程
大数据·git·elasticsearch
Elastic 中国社区官方博客1 天前
Jina Reranker v3:用于 SOTA 多语言检索 的 0.6B 列表式重排序器
大数据·人工智能·elasticsearch·搜索引擎·ai·jina
Dxy12393102161 天前
ES的DSL编写规则规则讲解
大数据·elasticsearch·搜索引擎