SpringDataElasticsearch扩展repository


前言

本文章可以直接复制粘贴,直接运行

步骤

1. maven依赖

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
        </dependency>

2.配置文件

我这边就简单配置下,自己的demo,就没有配置那么复杂。

search:
  elasticsearch:
    uris:
      - http://localhost:9200

3.创建一个接口继承 ElasticsearchRepository<T,ID>

代码如下:

@NoRepositoryBean
public interface CustomElasticsearchRepository<T, ID> extends ElasticsearchRepository<T, ID> {
    Page<T> searchPage(Pageable pageable, Query<?> esQuery);
}

4.实现CustomElasticsearchRepository,实现类继承 SimpleElasticsearchRepository

代码如下:

public class SimpleCustomElasticsearchRepository<T, ID> extends SimpleElasticsearchRepository<T, ID> implements CustomElasticsearchRepository<T, ID> {
    private final ElasticsearchRestTemplate elasticsearchRestTemplate;

    public SimpleCustomElasticsearchRepository(ElasticsearchEntityInformation<T, ID> metadata, ElasticsearchOperations elasticsearchOperations) {
        super(metadata, elasticsearchOperations);
        elasticsearchRestTemplate = (ElasticsearchRestTemplate) super.operations;
    }

	    @Override
    @SuppressWarnings("unchecked")
    public Page<T> searchPage(Pageable pageable, Query<?> esQuery) {
        NativeSearchQuery query = new NativeSearchQueryBuilder().withQuery(esQuery.getQuery()).withPageable(pageable).build();
        SearchHits<T> searchHits = operations.search(query, getEntityClass(), this.getIndexCoordinates());
        AggregatedPage<SearchHit<T>> page = SearchHitSupport.page(searchHits, query.getPageable());
        return (Page<T>) SearchHitSupport.unwrapSearchHits(page);
    }
 }

Query<?> query 使我自己封装的一个lambda条件查询器。具体实现形式可以参考Mybatis-plus 谁能拒绝现成的代码cv呢?毕竟开头都说了本文章直接可以cv运行,做人要说话算数
代码如下:

public interface Query<T> {
    String KEYWORD = ".keyword";

    QueryBuilder getQuery();
}
/**
 * 
 * must 查询 = mysql中 and
 * should查询 = mysql中 or
 *
 * @author dys
 * @date 2022/5/30 14:07
 */
public class BoolLambdaQuery<T> extends AbstractLambdaQuery<T> {

    private final BoolQueryBuilder query;

    public BoolLambdaQuery() {
        this.query = QueryBuilders.boolQuery();
    }

    public <R> BoolLambdaQuery<T> eq(SFunction<T, R> func, Object param) {
        query.must(QueryBuilders.termQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> orEq(SFunction<T, R> func, Object param) {
        query.should(QueryBuilders.termQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> noEq(SFunction<T, R> func, Object param) {
        query.mustNot(QueryBuilders.termQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> eqKw(SFunction<T, R> func, Object param) {
        query.must(QueryBuilders.termQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> orEqKw(SFunction<T, R> func, Object param) {
        query.should(QueryBuilders.termQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> noEqKw(SFunction<T, R> func, Object param) {
        query.mustNot(QueryBuilders.termQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> like(SFunction<T, R> func, Object param) {
        query.must(QueryBuilders.wildcardQuery(FunctionUtil.getName(func), "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> orLike(SFunction<T, R> func, Object param) {
        query.should(QueryBuilders.wildcardQuery(FunctionUtil.getName(func), "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> noLike(SFunction<T, R> func, Object param) {
        query.mustNot(QueryBuilders.wildcardQuery(FunctionUtil.getName(func), "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> likeKw(SFunction<T, R> func, Object param) {
        query.must(QueryBuilders.wildcardQuery(FunctionUtil.getName(func) + KEYWORD, "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> orLikeKw(SFunction<T, R> func, Object param) {
        query.should(QueryBuilders.wildcardQuery(FunctionUtil.getName(func) + KEYWORD, "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> noLikeKw(SFunction<T, R> func, Object param) {
        query.mustNot(QueryBuilders.wildcardQuery(FunctionUtil.getName(func) + KEYWORD, "*" + param + "*"));
        return this;
    }

    public <R> BoolLambdaQuery<T> in(SFunction<T, R> func, Object... param) {
        query.must(QueryBuilders.termsQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> orIn(SFunction<T, R> func, Object... param) {
        query.should(QueryBuilders.termsQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> noIn(SFunction<T, R> func, Object... param) {
        query.mustNot(QueryBuilders.termsQuery(FunctionUtil.getName(func), param));
        return this;
    }

    public <R> BoolLambdaQuery<T> in(SFunction<T, R> func, Collection<Object> collection) {
        query.must(QueryBuilders.termsQuery(FunctionUtil.getName(func), collection));
        return this;
    }

    public <R> BoolLambdaQuery<T> orIn(SFunction<T, R> func, Collection<Object> collection) {
        query.should(QueryBuilders.termsQuery(FunctionUtil.getName(func), collection));
        return this;
    }

    public <R> BoolLambdaQuery<T> noIn(SFunction<T, R> func, Collection<Object> collection) {
        query.mustNot(QueryBuilders.termsQuery(FunctionUtil.getName(func), collection));
        return this;
    }

    public <R> BoolLambdaQuery<T> inKw(SFunction<T, R> func, Object... param) {
        query.must(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> orInKw(SFunction<T, R> func, Object... param) {
        query.should(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> noInKw(SFunction<T, R> func, Object... param) {
        query.mustNot(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, param));
        return this;
    }

    public <R> BoolLambdaQuery<T> inKw(SFunction<T, R> func, Collection<Object> collection) {
        query.must(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, collection));
        return this;
    }

    public <R> BoolLambdaQuery<T> orInKw(SFunction<T, R> func, Collection<Object> collection) {
        query.should(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, collection));
        return this;
    }

    public <R> BoolLambdaQuery<T> noInKw(SFunction<T, R> func, Collection<Object> collection) {
        query.mustNot(QueryBuilders.termsQuery(FunctionUtil.getName(func) + KEYWORD, collection));
        return this;
    }

    public BoolLambdaQuery<T> must(Query<T> query) {
        this.query.must(query.getQuery());
        return this;
    }

    public BoolLambdaQuery<T> should(Query<T> query) {
        this.query.must(query.getQuery());
        return this;
    }

    public BoolLambdaQuery<T> mustNot(Query<T> query) {
        this.query.mustNot(query.getQuery());
        return this;
    }

    @Override
    public QueryBuilder getQuery() {
        return this.query;
    }
@FunctionalInterface
public interface SFunction<T, R> extends Function<T, R>, Serializable {
}
public class FunctionUtil {
    private static final String GET = "get";

    private static final String WRITE_REPLACE = "writeReplace";

    public static <T, R> String getName(SFunction<T, R> func) {
        // 直接调用writeReplace
        try {
            Method writeReplace = func.getClass().getDeclaredMethod(WRITE_REPLACE);
            writeReplace.setAccessible(true);
            //反射调用
            Object sl = writeReplace.invoke(func);
            java.lang.invoke.SerializedLambda serializedLambda = (java.lang.invoke.SerializedLambda) sl;
            return StringUtil.stringStartTrim(serializedLambda.getImplMethodName(), GET);
        } catch (NoSuchMethodException | InvocationTargetException | IllegalAccessException e) {
            throw new RuntimeException(e);
        }
    }
}

5.Application上增加注释,指定SimpleCustomElasticsearchRepository

@SpringBootApplication
@EnableElasticsearchRepositories(repositoryBaseClass = SimpleCustomElasticsearchRepository.class)
public class EsDemoApplication {

    public static void main(String[] args) {
        SpringApplication.run(EsDemoApplication.class, args);
    }

}

6.其他Repository接口继承CustomElasticsearchRepository

public interface CustomerInfoRepository extends CustomElasticsearchRepository<CustomerInfo, Long> {
}

示例结束,如果需要其他方法进行封装的话可以直接添加。
例如,手动扩展Query的功能,将排序,分页封装进去,lambda函数话编程就写到的多爽

        BooLambdaQuery<CustomerInfo> lambdaQuery = new BooLambdaQuery<>();
        lambdaQuery.MatchAll()
                .pageBuilder(DefPageQuery.class)
                .sort(CustomerInfo::getId, SortOrder.DESC)
                .page(0)
                .size(5);
        Page<CustomerInfo> customerInfoPage = customerInfoService.pageByQuery(lambdaQuery);
        customerInfoPage.getContent().forEach(System.out::println);

这样看着多舒服,还能一个分页接口实现elasticsearch中的三种分页。
皮一下 这个不发~~

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
Elasticsearch是一个基于Lucene的分布式全文搜索引擎,它具备高性能、可扩展性和易用性。ElasticsearchRepositorySpring Data Elasticsearch提供的一个接口,用于简化与Elasticsearch的交互。 使用ElasticsearchRepository,我们可以通过定义接口和方法来快速实现对Elasticsearch的基本操作,比如保存、查找、删除和更新文档。首先,我们需要创建一个继承自ElasticsearchRepository的接口,并指定实体类型和文档ID的类型。接口中的方法名采用了一种特殊的命名规则,用来表示不同的操作。 比如,我们可以定义一个方法来查找指定ID的文档: ``` Optional<User> findById(String id); ``` 或者,我们可以定义一个方法来根据特定条件进行查询: ``` List<User> findByAgeGreaterThan(int age); ``` 使用ElasticsearchRepository,我们只需要定义接口和方法,而无需编写实现代码。Spring Data Elasticsearch会根据方法名和参数,自动帮我们生成相应的查询语句,并将查询结果转化为我们定义的实体类型。 此外,ElasticsearchRepository还提供了一些其他的方法,用于处理分页、排序和聚合等操作。通过这些方法,我们可以轻松地进行复杂的数据查询和分析。 总之,使用ElasticsearchRepository可以极大地简化与Elasticsearch的集成开发。它提供了一种简洁而高效的方式来操作Elasticsearch,并且与Spring Data系列的其他组件相互融合,使得开发更加方便和灵活。
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值