这几天写了一个关于es的工具类,主要封装了业务中常用es的常用方法。

  本文中使用到的elasticsearch版本6.7,但实际上也支持es7.x以上版本,因为主要是对springboot提供的:ElasticsearchRestTemplate 提供的API做的二次封装。目的是:让不懂es的开发人员新手也能轻松上手。

一、概述

整个工程分为es-api与es-server。

es-api为对外公共jar,包含了es映射实体;

es-server包含了具体的es工具类与Repository接口(下文会提到)。并通过necos配置中心统一管理es配置参数。

外部业务模块可引入es-api jar maven依赖,由Jar提供的入口,通过httpClient或feign调用(springcloud分布式项目)到es-server服务上的es工具类,得到需要的数据。

二、使用

这里仅以springcloud分布式项目简单为例

业务方用户服务user 引入es-api maven

/**
* @author: shf
* description: es-server feign接口
*/
public interface EsServerClient {
@PostMapping(value = "/queryList", produces = {"application/json"})
public <T> List<T> queryList(@RequestBody T t);
}

在user服务中创建feignClient继承自es-api中的EsServerClient

@FeignClient(contextId = "esFeignClient", name = "es-server")
public interface EsFeignClient extends EsServerClient {
}

在user服务的代码中即可调用

@AutoWired
public EsFeignClient esFeignClient; public void test() {
//.......如业务方Dto与es映射实体转换 等省略
//....
EmployeeEs employee = new EmployeeEs();
List queryList = Stream.of(employee.new QueryRelation<String>("张三", EntityEs.SHOULD, 5F), employee.new QueryRelation<String>("李四", EntityEs.SHOULD, 20F)).collect(Collectors.toList());
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, queryList).put(EmployeeEs::getUserAge, employee.new RangeRelation(20, EntityEs.GTE, null, null, EntityEs.MUST)));
//排序查询
employee.setOrderMap(new EsMapUtil().put(EmployeeEs::getUserId, SortOrder.DESC));
List<EmployeeEs> employeeEs = esFeignClient.queryList(employee);
//.....employeeEs与业务方Dto转换
}

三、具体实现

es-api 引入es依赖

<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-elasticsearch</artifactId>
<exclusions>
<exclusion>
<groupId>org.elasticsearch.client</groupId>
<artifactId>transport</artifactId>
</exclusion>
<exclusion>
<groupId>org.elasticsearch.client</groupId>
<artifactId>elasticsearch-rest-high-level-client</artifactId>
</exclusion>
</exclusions>
</dependency>
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>transport</artifactId>
<version>6.7.0</version>
</dependency>
<dependency>
<groupId>org.elasticsearch.client</groupId>
<artifactId>elasticsearch-rest-high-level-client</artifactId>
<version>6.7.0</version>
</dependency>

排除后重新引入对应的Es版本6.7,避免因版本不一致导致的一些坑。

es-server 服务 application.yml配置es

spring:
elasticsearch:
rest:
#ES的连接地址,多个地址用逗号分隔
uris: localhost:9200
username: kibana
password: pass
#连接超时时间
connection-timeout: 1000
#读取超时时间
read-timeout: 1000

一、映射实体

1、与ES mapping结构对应的映射实体:EmployeeEs

说明:

①设置的字段与该es对应的该索引完全对应,不存在多余字段。

②项目中引入了spring-boot-starter-data-elasticsearch,所以可直接使用注解形式设置索引信息与mapping结构信息,详见示例

③@JsonIgnoreProperties({"orderMap","pageNumber","pageSize","highlightFields","preTags","postTags","fieldQueryMap","scrollId","aggregationMap","multiLayerQueryList"})

作用:在保存文档到es时忽略父类EntityEs中的功能性字段,下文会提到

④@EsRepository(EmployeeEsRepository.class)

作用:通过注解过去该映射对应的Repository接口

/**
* 员工对象
* <p>
* 注解:@Document用来声明Java对象与ElasticSearch索引的关系 indexName 索引名称 type 索引类型 shards 主分区数量,默认5
* replicas 副本分区数量,默认1 createIndex 索引不存在时,是否自动创建索引,默认true
*/
@Setter
@Getter
@Builder
@AllArgsConstructor
@NoArgsConstructor
@EsRepository(EmployeeEsRepository.class)
@JsonIgnoreProperties({"orderMap","pageNumber","pageSize","highlightFields","preTags","postTags","fieldQueryMap","scrollId","aggregationMap","multiLayerQueryList"})
@Document(indexName = "employee_index", type = "employee_type", shards = 1, replicas = 0, createIndex = true)
public class EmployeeEs extends EntityEs { @Id
@Field(type = FieldType.Keyword)
private Long userId; //@Field(type = FieldType.Text, analyzer = "ik_max_word")
@MultiField(mainField = @Field(type = FieldType.Text, analyzer = "ik_max_word"), otherFields = @InnerField(suffix = "trueName", type = FieldType.Keyword))
private String userName; @Field(type = FieldType.Keyword)
private String userCode; @Field(type = FieldType.Integer)
private Integer userAge; @Field(type = FieldType.Keyword)
private String userMobile; @Field(type = FieldType.Date)
private Date birthDay; @Field(type = FieldType.Keyword)
private String userSex; @Field(type = FieldType.Text, analyzer = "ik_max_word")
private String remarks;
}

2、Repository接口:EmployeeEsRepository

/**
* @author: shf
* description: 可根据映射实体设置自动生成mapping结构;支持bean的增删改查操作
* date: 2022/2/23 10:47
*/
@Component
public interface EmployeeEsRepository extends CrudRepository<EmployeeEs,Long> {
}

二、功能性实体类:EntityEs

说明:

①所有字段非es索引中的mapping属性字段,均为功能性字段,如排序、高亮、分页设置和一些常量设置等,详见贴码

②所有es映射实体类均需继承该实体

/**
* @author: shf description: 功能性字段(非mapping结构字段)
* date: 2022/3/1 15:07
*/
@Data
public class EntityEs { /**
* 组合多查询常量
*/
/**
* 文档 必须 匹配这些条件才能被查询到。相当于sql中的and
*/
public static String MUST = "must"; /**
* 文档 必须不 匹配这些条件才能被查询到。相当于sql中的 not
*/
public static String MUST_NOT = "must_not"; /**
* 如果满足这些语句中的任意语句,将增加 _score ,否则,无任何影响。它们主要用于修正每个文档的相关性得分。相当于sql中的or
*/
public static String SHOULD = "should"; /**
* 必须 匹配,但它以不评分、过滤模式来进行。这些语句对评分没有贡献,只是根据过滤标准来排除或包含文档
*/
public static String FILTER = "filter"; /**
* 至少匹配一项should子句
*/
public static String MINIMUM_SHOULD_MATCH = "minimum_should_match"; /**
* 多字段排序查询
*/
public EsMapUtil orderMap; /**
* 分页查询
*/
public Integer pageNumber; public Integer pageSize; /**
* 游标分页ID
*/
public String scrollId; /**
* 游标分页ID有效期 单位:毫秒
*/
public static Long scrollIdExpireTime = 1000 * 60 * 2L; /**
* 游标分页ID最小有效期 单位:毫秒
*/
public static Long scrollIdMinExpireTime = 1000L; /**
* 高亮查询
*/
public List<String> highlightFields; public String preTags; public String postTags; /**
* 字段查询
*/
public EsMapUtil fieldQueryMap; /**
* 聚合查询,当前只支持单个字段分组聚合count与sum,只针对keyword类型字段有效
*/
public EsMapUtil aggregationMap; public static String COUNT = "count";
public static String SUM = "sum"; /**
* 多层(bool)查询
*/
public List multiLayerQueryList; /**
* 范围查询常量
*/
public static String GT = "gt";
public static String GTE = "gte";
public static String LT = "lt";
public static String LTE = "lte"; @Data
public class RangeRelation<T> { //String fieldKey; T fieldMinValue; String fieldMinMode; T fieldMaxValue; String fieldMaxMode; String queryMode; public RangeRelation(T fieldMinValue, String fieldMinMode, T fieldMaxValue, String fieldMaxMode, String queryMode) {
this.fieldMinValue = fieldMinValue;
this.fieldMinMode = fieldMinMode;
this.fieldMaxValue = fieldMaxValue;
this.fieldMaxMode = fieldMaxMode;
this.queryMode = queryMode;
}
} @Data
public class QueryRelation<T> { T fieldValue; String queryMode; Float boostValue; public QueryRelation(T fieldValue, String queryMode) {
this.fieldValue = fieldValue;
this.queryMode = queryMode;
} public QueryRelation(T fieldValue, String queryMode, Float boostValue) {
this.fieldValue = fieldValue;
this.queryMode = queryMode;
this.boostValue = boostValue;
}
} @Data
public class MultiLayerRelation { String queryMode; EsMapUtil map; List<EntityEs.MultiLayerRelation> multiLayerList; public MultiLayerRelation(String queryMode, EsMapUtil map) {
this.queryMode = queryMode;
this.map = map;
} public MultiLayerRelation(String queryMode, EsMapUtil map, List<MultiLayerRelation> multiLayerList) {
this.queryMode = queryMode;
this.map = map;
this.multiLayerList = multiLayerList;
}
}
}

三、小工具:EsMapUtil

说明:封装了一个map工具,编码简洁链式调用,应用了方法引用特性避免了字符串硬编码造成单词拼错的情况。

/**
* @author: shf description: 函数式接口 便于方法引用获取实体字段名称
* date: 2022/3/4 13:41
*/
@FunctionalInterface
public interface IGetterFunction<T> extends Serializable{
Object get(T source);
}

/**
* @author: shf
* description
* date: 2019/11/13 18:30
*/
public class EsMapUtil extends LinkedHashMap<String, Object> { public <T> EsMapUtil put(IGetterFunction<T> fn, Object value) {
String key = getFieldName(fn);
super.put(key, value);
return this;
} public <T> EsMapUtil putStr(String key, Object value) {
super.put(key, value);
return this;
} private static Map<Class, SerializedLambda> CLASS_LAMDBA_CACHE = new ConcurrentHashMap<>(); /***
* 转换方法引用为属性名
* @param fn
* @return
*/
public <T> String getFieldName(IGetterFunction<T> fn) {
SerializedLambda lambda = getSerializedLambda(fn);
String methodName = lambda.getImplMethodName();
String prefix = null;
if (methodName.startsWith("get")) {
prefix = "get";
}
// 截取get之后的字符串并转换首字母为小写
return toLowerCaseFirstOne(methodName.replace(prefix, ""));
} /**
* 首字母转小写
*
* @param s
*/
public String toLowerCaseFirstOne(String s) {
if (Character.isLowerCase(s.charAt(0))) {
return s;
} else {
return (new StringBuilder()).append(Character.toLowerCase(s.charAt(0))).append(s.substring(1)).toString();
}
} public static SerializedLambda getSerializedLambda(Serializable fn) {
SerializedLambda lambda = CLASS_LAMDBA_CACHE.get(fn.getClass());
if (lambda == null) {
try {
Method method = fn.getClass().getDeclaredMethod("writeReplace");
method.setAccessible(Boolean.TRUE);
lambda = (SerializedLambda) method.invoke(fn);
CLASS_LAMDBA_CACHE.put(fn.getClass(), lambda);
} catch (Exception e) {
e.printStackTrace();
}
}
return lambda;
}
}

四、Es通用工具类EsService

/**
* @author: shf description: es工具类,支持:分页(支持游标分页)、高亮(支持自定义标签)、范围查找、bool组合查询、多层bool套bool、多字段排序、加权重、聚合、二级字段查询
* date: 2022/2/23 10:54
*/
@Component
@Slf4j
public class EsService {
@Autowired
private ElasticsearchRestTemplate restTemplate; @Autowired
private ApplicationContext context; /**
* 判断索引是否存在
*
* @return boolean
*/
public <T> boolean indexExists(Class<T> clazz) {
return restTemplate.indexExists(clazz);
} /**
* 判断索引是否存在
*
* @param indexName 索引名称
* @return boolean
*/
public boolean indexExists(String indexName) {
return restTemplate.indexExists(indexName);
} /**
* 创建索引(推荐使用:因为Java对象已经通过注解描述了Setting和Mapping)
*
* @return boolean
*/
public <T> boolean indexCreate(Class<T> clazz) {
boolean createFlag = restTemplate.createIndex(clazz);
boolean mappingFlag = restTemplate.putMapping(clazz);
return createFlag && mappingFlag; } /**
* 索引删除
*
* @param indexName 索引名称
* @return boolean
*/
public boolean indexDelete(String indexName) {
return restTemplate.deleteIndex(indexName);
} /**
* 新增数据
*
* @param bean 数据对象
*/
public <T> void saveBean(T bean) {
EsRepository esRepositoryAnno = bean.getClass().getAnnotation(EsRepository.class);
CrudRepository crudRepository = (CrudRepository) context.getBean(esRepositoryAnno.value());
crudRepository.save(bean);
} /**
* 批量新增数据
*
* @param list 数据集合
*/
public <T> void saveList(List<T> list) {
if (CollectionUtils.isEmpty(list)) {
return;
}
EsRepository esRepositoryAnno = list.get(0).getClass().getAnnotation(EsRepository.class);
CrudRepository crudRepository = (CrudRepository) context.getBean(esRepositoryAnno.value());
crudRepository.saveAll(list);
} /**
* 根据对象删除数据,主键ID不能为空
*
* @param bean 对象
*/
public <T> void deleteByBean(T bean) {
EsRepository esRepositoryAnno = bean.getClass().getAnnotation(EsRepository.class);
CrudRepository crudRepository = (CrudRepository) context.getBean(esRepositoryAnno.value());
crudRepository.delete(bean);
} /**
* 根据对象集合,批量删除
*
* @param beanList 对象集合
*/
public <T> void deleteAllByBeanList(List<T> beanList) {
if (CollectionUtils.isEmpty(beanList)) {
return;
}
EsRepository esRepositoryAnno = beanList.get(0).getClass().getAnnotation(EsRepository.class);
CrudRepository crudRepository = (CrudRepository) context.getBean(esRepositoryAnno.value());
crudRepository.deleteAll(beanList);
} /**
* 删除所有
*/
public <T> void deleteAll(T bean) {
EsRepository esRepositoryAnno = bean.getClass().getAnnotation(EsRepository.class);
CrudRepository crudRepository = (CrudRepository) context.getBean(esRepositoryAnno.value());
crudRepository.deleteAll();
} /**
* 修改数据
*
* @param t 修改数据对象,ID不能为空
*/
public <T> boolean updateByBean(T t) throws IllegalAccessException {
Class clazz = t.getClass();
Field[] Fields = clazz.getDeclaredFields();
String beanId = null;
String beanIdName = null;
for (Field f : Fields) {
f.setAccessible(true);
if (f.isAnnotationPresent(org.springframework.data.annotation.Id.class)) {
beanId = String.valueOf(f.get(t));
beanIdName = f.getName();
}
}
if (StringUtils.isBlank(beanId)) {
log.warn("id不能为空");
return false;
}
if (Objects.isNull(restTemplate.queryForObject(GetQuery.getById(beanId), clazz))) {
log.warn("该文档不存在");
return false;
}
Document annotation = (Document) clazz.getAnnotation(Document.class);
UpdateRequest updateRequest = new UpdateRequest();
//冲突重试
updateRequest.retryOnConflict(1);
updateRequest.doc(JSON.toJSONString(t), XContentType.JSON);
//默认是_id来路由的,用来路由到不同的shard,会对这个值做hash,然后映射到shard。所以分片
updateRequest.routing(beanId);
UpdateQuery query = new UpdateQueryBuilder().withIndexName(annotation.indexName()).withType(annotation.type()).withId(beanId)
.withDoUpsert(false)//不加默认false。true表示更新时不存在就插入
.withClass(clazz).withUpdateRequest(updateRequest).build();
UpdateResponse updateResponse = restTemplate.update(query);
if (!Objects.equals(updateResponse.getShardInfo().getSuccessful(), 1)) {
return false;
}
return true;
} /**
* 根据bean ID 查询
*
* @param beanId
* @param clazz
* @param <T>
*/
public <T> T queryBeanById(String beanId, Class<T> clazz) {
return restTemplate.queryForObject(GetQuery.getById(beanId), clazz);
} /**
* 数据查询,返回List
*
* @return List<T>
*/
public <T> List<T> queryList(T t) throws IllegalAccessException, NoSuchFieldException {
BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
Class clazz = (Class) t.getClass();
Map<String, Object> queryMap = (LinkedHashMap) clazz.getField("fieldQueryMap").get(t);
Map<String, String> orderMap = (LinkedHashMap) clazz.getField("orderMap").get(t);
List<String> highlightFields = (List<String>) clazz.getField("highlightFields").get(t);
String preTags = StringUtils.isNotBlank((String) clazz.getField("preTags").get(t)) ? (String) clazz.getField("preTags").get(t) : "<em>";
String postTags = StringUtils.isNotBlank((String) clazz.getField("postTags").get(t)) ? (String) clazz.getField("postTags").get(t) : "</em>";
List<EntityEs.MultiLayerRelation> multiLayerQueryList = (List<EntityEs.MultiLayerRelation>) clazz.getField("multiLayerQueryList").get(t); String beanIdName = null;
Field[] Fields = clazz.getDeclaredFields();
for (Field f : Fields) {
f.setAccessible(true);
if (f.isAnnotationPresent(org.springframework.data.annotation.Id.class)) {
beanIdName = f.getName();
break;
}
}
//构建组合查询(支持权重)
getFieldQueryBuilder(boolQueryBuilder, clazz, queryMap);
//处理多层bool查询
getNestQueryBuilder(boolQueryBuilder, clazz, multiLayerQueryList); log.info("打印语句:{}", boolQueryBuilder.toString());
NativeSearchQueryBuilder nativeSearchQueryBuilder = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder);
//支持多字段排序查询
getOrderBuilder(beanIdName, orderMap, nativeSearchQueryBuilder);
//支持多字段高亮查询并可自定义高亮标签规则
getHighLightBuilder(highlightFields, preTags, postTags, nativeSearchQueryBuilder); if (CollectionUtils.isEmpty(highlightFields)) {
return restTemplate.queryForList(nativeSearchQueryBuilder.build(), clazz);
} else {
nativeSearchQueryBuilder.withPageable(PageRequest.of(0, 10000));
ScrolledPage page = restTemplate.startScroll(EntityEs.scrollIdMinExpireTime, nativeSearchQueryBuilder.build(), clazz, new SearchResultMapper() {
@Override
public <T> AggregatedPage<T> mapResults(SearchResponse response, Class<T> clazz, Pageable pageable) {
List<T> chunk = new ArrayList<>();
for (SearchHit searchHit : response.getHits()) {
if (response.getHits().getHits().length <= 0) {
return null;
}
try {
T t = (T) JSON.parseObject(searchHit.getSourceAsString(), clazz);
for (String fieldName : highlightFields) {
Field f = clazz.getDeclaredField(fieldName);
HighlightField highlightField = searchHit.getHighlightFields().get(fieldName);
if (highlightField != null) {
f.setAccessible(true);
f.set(t, highlightField.fragments()[0].toString());
}
}
chunk.add(t);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}
if (chunk.size() > 0) {
return new AggregatedPageImpl<>((List<T>) chunk);
}
return null;
} @Override
public <T> T mapSearchHit(SearchHit searchHit, Class<T> type) {
return null;
}
});
return page.toList();
}
} /**
* 分页查询
*
* @param t
* @param <T>
*/
public <T> List<T> queryPage(T t) throws IllegalAccessException, NoSuchFieldException {
BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
Class clazz = (Class) t.getClass(); Map<String, Object> queryMap = (LinkedHashMap) clazz.getField("fieldQueryMap").get(t);
Map<String, String> orderMap = (LinkedHashMap) clazz.getField("orderMap").get(t);
List<String> highlightFields = (List<String>) clazz.getField("highlightFields").get(t);
String preTags = StringUtils.isNotBlank((String) clazz.getField("preTags").get(t)) ? (String) clazz.getField("preTags").get(t) : "<em>";
String postTags = StringUtils.isNotBlank((String) clazz.getField("postTags").get(t)) ? (String) clazz.getField("postTags").get(t) : "</em>";
Integer pageNumber = !Objects.isNull((Integer) clazz.getField("pageNumber").get(t)) ? (Integer) clazz.getField("pageNumber").get(t) : 0;
Integer pageSize = !Objects.isNull((Integer) clazz.getField("pageSize").get(t)) ? (Integer) clazz.getField("pageSize").get(t) : 10;
List<EntityEs.MultiLayerRelation> multiLayerQueryList = (List<EntityEs.MultiLayerRelation>) clazz.getField("multiLayerQueryList").get(t); String beanIdName = null;
Field[] Fields = clazz.getDeclaredFields();
for (Field f : Fields) {
f.setAccessible(true);
if (f.isAnnotationPresent(org.springframework.data.annotation.Id.class)) {
beanIdName = f.getName();
break;
}
}
//构建组合查询(支持权重)
getFieldQueryBuilder(boolQueryBuilder, clazz, queryMap);
//处理多层bool查询
getNestQueryBuilder(boolQueryBuilder, clazz, multiLayerQueryList);
NativeSearchQueryBuilder nativeSearchQueryBuilder = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder);
//支持多字段排序查询
getOrderBuilder(beanIdName, orderMap, nativeSearchQueryBuilder);
//支持多字段高亮查询并可自定义高亮标签规则
getHighLightBuilder(highlightFields, preTags, postTags, nativeSearchQueryBuilder);
//分页查询
nativeSearchQueryBuilder.withPageable(PageRequest.of(pageNumber, pageSize)); AggregatedPage page = null;
if (CollectionUtils.isEmpty(highlightFields)) {
page = restTemplate.queryForPage(nativeSearchQueryBuilder.build(), clazz);
} else {
page = restTemplate.queryForPage(nativeSearchQueryBuilder.build(), clazz, new SearchResultMapper() {
@Override
public <T> AggregatedPage<T> mapResults(SearchResponse response, Class<T> clazz, Pageable pageable) {
List<T> chunk = new ArrayList<>();
for (SearchHit searchHit : response.getHits()) {
if (response.getHits().getHits().length <= 0) {
return null;
}
try {
T t = (T) JSON.parseObject(searchHit.getSourceAsString(), clazz);
for (String fieldName : highlightFields) {
Field f = clazz.getDeclaredField(fieldName);
HighlightField highlightField = searchHit.getHighlightFields().get(fieldName);
if (highlightField != null) {
f.setAccessible(true);
f.set(t, highlightField.fragments()[0].toString());
}
}
chunk.add(t);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}
if (chunk.size() > 0) {
return new AggregatedPageImpl<>((List<T>) chunk);
} return null;
} @Override
public <T> T mapSearchHit(SearchHit searchHit, Class<T> type) {
return null;
}
});
}
// 总记录数
long totalElements = page.getTotalElements();
// 总页数
int totalPages = page.getTotalPages();
// 当前页号
//int currentPage = page.getPageable().getPageNumber();
// 当前页数据集
List<T> beanList = page.toList();
List<T> content = page.getContent();
System.out.println(beanList);
//TODO 根据项目中的分页封装类将以上数据设置后返回即可
return Lists.newArrayList();
} /**
* 游标分页
*
* @param t
* @param <T>
*/
public <T> List<T> queryScrollPage(T t) throws IllegalAccessException, NoSuchFieldException {
BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
Class clazz = (Class) t.getClass();
String scrollId = (String) clazz.getField("scrollId").get(t);
if (StringUtils.isNotBlank(scrollId)) {
ScrolledPage page = restTemplate.continueScroll(scrollId, EntityEs.scrollIdExpireTime, clazz);
// 总记录数
long totalElements = page.getTotalElements();
// 总页数
//int totalPages = page.getTotalPages();
// 当前页号
//int currentPage = page.getPageable().getPageNumber();
// 当前页数据集
List<T> beanSet = page.toList();
List<T> content = page.getContent();
System.out.println("page.getScrollId:" + page.getScrollId());
System.out.println(beanSet);
//TODO 根据项目中的分页封装类将以上数据设置后返回即可
return Lists.newArrayList();
}
Map<String, Object> queryMap = (LinkedHashMap) clazz.getField("fieldQueryMap").get(t);
Map<String, String> orderMap = (LinkedHashMap) clazz.getField("orderMap").get(t);
List<String> highlightFields = (List<String>) clazz.getField("highlightFields").get(t);
String preTags = StringUtils.isNotBlank((String) clazz.getField("preTags").get(t)) ? (String) clazz.getField("preTags").get(t) : "<em>";
String postTags = StringUtils.isNotBlank((String) clazz.getField("postTags").get(t)) ? (String) clazz.getField("postTags").get(t) : "</em>";
Integer pageNumber = !Objects.isNull((Integer) clazz.getField("pageNumber").get(t)) ? (Integer) clazz.getField("pageNumber").get(t) : 0;
Integer pageSize = !Objects.isNull((Integer) clazz.getField("pageSize").get(t)) ? (Integer) clazz.getField("pageSize").get(t) : 10;
List<EntityEs.MultiLayerRelation> multiLayerQueryList = (List<EntityEs.MultiLayerRelation>) clazz.getField("multiLayerQueryList").get(t); String beanIdName = null;
Field[] Fields = clazz.getDeclaredFields();
for (Field f : Fields) {
f.setAccessible(true);
if (f.isAnnotationPresent(org.springframework.data.annotation.Id.class)) {
beanIdName = f.getName();
break;
}
}
//构建组合查询(支持权重)
getFieldQueryBuilder(boolQueryBuilder, clazz, queryMap);
//处理多层bool查询
getNestQueryBuilder(boolQueryBuilder, clazz, multiLayerQueryList);
NativeSearchQueryBuilder nativeSearchQueryBuilder = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder);
//支持多字段排序查询
getOrderBuilder(beanIdName, orderMap, nativeSearchQueryBuilder);
//支持多字段高亮查询并可自定义高亮标签规则
getHighLightBuilder(highlightFields, preTags, postTags, nativeSearchQueryBuilder);
//分页查询
nativeSearchQueryBuilder.withPageable(PageRequest.of(pageNumber, pageSize)); ScrolledPage page = restTemplate.startScroll(EntityEs.scrollIdExpireTime, nativeSearchQueryBuilder.build(), clazz, new SearchResultMapper() {
@Override
public <T> AggregatedPage<T> mapResults(SearchResponse response, Class<T> clazz, Pageable pageable) {
List<T> chunk = new ArrayList<>();
for (SearchHit searchHit : response.getHits()) {
if (response.getHits().getHits().length <= 0) {
return null;
}
try {
T t = (T) JSON.parseObject(searchHit.getSourceAsString(), clazz);
for (String fieldName : highlightFields) {
Field f = clazz.getDeclaredField(fieldName);
HighlightField highlightField = searchHit.getHighlightFields().get(fieldName);
if (highlightField != null) {
f.setAccessible(true);
f.set(t, highlightField.fragments()[0].toString());
}
}
chunk.add(t);
} catch (IllegalAccessException e) {
e.printStackTrace();
} catch (NoSuchFieldException e) {
e.printStackTrace();
}
}
if (chunk.size() > 0) {
return new AggregatedPageImpl<>((List<T>) chunk);
}
return null;
} @Override
public <T> T mapSearchHit(SearchHit searchHit, Class<T> type) {
return null;
}
}); // 总记录数
long totalElements = page.getTotalElements();
// 总页数
//int totalPages = page.getTotalPages();
// 当前页号
//int currentPage = page.getPageable().getPageNumber();
// 当前页数据集
List<T> beanSet = page.toList();
List<T> content = page.getContent();
System.out.println("page.getScrollId:" + page.getScrollId());
System.out.println(beanSet);
//TODO 根据项目中的分页封装类将以上数据设置后返回即可
return Lists.newArrayList();
} /**
* 聚合查询
*
* @param t
* @param <T>
*/
public <T> Map queryForAggregation(T t) throws IllegalAccessException, NoSuchFieldException {
BoolQueryBuilder boolQueryBuilder = QueryBuilders.boolQuery();
Class clazz = (Class) t.getClass(); Map<String, Object> queryMap = (LinkedHashMap) clazz.getField("fieldQueryMap").get(t);
Integer pageNumber = !Objects.isNull(clazz.getField("pageNumber").get(t)) ? (Integer) clazz.getField("pageNumber").get(t) : 0;
Integer pageSize = !Objects.isNull(clazz.getField("pageSize").get(t)) ? (Integer) clazz.getField("pageSize").get(t) : 1;
Map<String, String> aggregationMap = (LinkedHashMap) clazz.getField("aggregationMap").get(t);
List<EntityEs.MultiLayerRelation> multiLayerQueryList = (List<EntityEs.MultiLayerRelation>) clazz.getField("multiLayerQueryList").get(t); //构建组合查询(支持权重)
getFieldQueryBuilder(boolQueryBuilder, clazz, queryMap);
//处理多层bool查询
getNestQueryBuilder(boolQueryBuilder, clazz, multiLayerQueryList);
NativeSearchQueryBuilder nativeSearchQueryBuilder = new NativeSearchQueryBuilder().withQuery(boolQueryBuilder);
//分页查询
nativeSearchQueryBuilder.withPageable(PageRequest.of(pageNumber, pageSize)); // 创建聚合查询条件
String aggKey = null;
String aggValue = null;
for (Map.Entry<String, String> entry : aggregationMap.entrySet()) {
aggKey = entry.getKey();
aggValue = entry.getValue();
if (Objects.equals(aggValue, EntityEs.COUNT)) {
TermsAggregationBuilder agg = AggregationBuilders.terms(aggKey).field(aggKey);
nativeSearchQueryBuilder.addAggregation(agg);
break;
} else if (Objects.equals(aggValue, EntityEs.SUM)) {
SumAggregationBuilder agg = AggregationBuilders.sum(aggKey).field(aggKey);
nativeSearchQueryBuilder.addAggregation(agg);
break;
}
}
AggregatedPage page = restTemplate.queryForPage(nativeSearchQueryBuilder.build(), clazz);
// 取出聚合结果
Aggregations entitiesAggregations = page.getAggregations();
Map<String, Object> retMap = new HashMap<>();
if (Objects.equals(aggValue, EntityEs.COUNT)) {
Terms terms = (Terms) entitiesAggregations.asMap().get(aggKey);
// 遍历取出聚合字段列的值,与对应的数量
for (Terms.Bucket bucket : terms.getBuckets()) {
// 聚合字段列的值
String keyAsString = bucket.getKeyAsString();
// 聚合字段对应的数量
long docCount = bucket.getDocCount();
log.info("keyAsString={},value={}", keyAsString, docCount);
retMap.put(keyAsString, docCount);
}
} else if (Objects.equals(aggValue, EntityEs.SUM)) {
Aggregation aggregation = entitiesAggregations.get(aggKey);
retMap.put(aggregation.getName(), ((ParsedSum) aggregation).getValue());
}
// 当前页数据集
List<T> beanList = page.toList();
System.out.println("当前页数据集:" + beanList);
return retMap;
} /**
* 根据自定义查询条件批量查询
*
* @param searchQuery
* @param clazz
* @param <T>
*/
public <T> List<T> queryListBySearchQuery(SearchQuery searchQuery, Class<T> clazz) {
return restTemplate.queryForList(searchQuery, clazz);
} /*------------------------------------------- private 私有方法 ----------------------------------------------*/ private void getNestQueryBuilder(BoolQueryBuilder boolQueryBuilder, Class clazz, List<EntityEs.MultiLayerRelation> multiLayerQueryList) throws NoSuchFieldException {
if (!CollectionUtils.isEmpty(multiLayerQueryList)) {
for (EntityEs.MultiLayerRelation r : multiLayerQueryList) {
BoolQueryBuilder nestBoolQuery = QueryBuilders.boolQuery();
EsMapUtil nestMap = r.getMap();
getFieldQueryBuilder(nestBoolQuery, clazz, nestMap);
if (Objects.equals(r.getQueryMode(), EntityEs.MUST)) {
boolQueryBuilder.must(nestBoolQuery);
} else if (Objects.equals(r.getQueryMode(), EntityEs.SHOULD)) {
boolQueryBuilder.should(nestBoolQuery);
} else if (Objects.equals(r.getQueryMode(), EntityEs.FILTER)) {
boolQueryBuilder.filter(nestBoolQuery);
} else if (Objects.equals(r.getQueryMode(), EntityEs.MUST_NOT)) {
boolQueryBuilder.mustNot(nestBoolQuery);
}
//递归嵌套
if (!CollectionUtils.isEmpty(r.getMultiLayerList())) {
//处理多层bool查询
getNestQueryBuilder(nestBoolQuery, clazz, r.getMultiLayerList());
}
}
}
} /**
* 构建组合查询(支持权重)
*
* @param boolQueryBuilder
* @param clazz
* @param queryMap
*/
private void getFieldQueryBuilder(BoolQueryBuilder boolQueryBuilder, Class clazz, Map<String, Object> queryMap) throws NoSuchFieldException {
if (queryMap != null && queryMap.size() > 0) {
for (Map.Entry<String, Object> entry : queryMap.entrySet()) {
String k = entry.getKey();
List vList = new ArrayList();
if (entry.getValue() instanceof List) {
vList = (ArrayList) entry.getValue();
} else {
vList.add(entry.getValue());
}
FieldType type = null;
if (k.indexOf(".") == -1) {
Field f = clazz.getDeclaredField(k);
if (f.isAnnotationPresent(org.springframework.data.elasticsearch.annotations.Field.class)) {
type = f.getAnnotation(org.springframework.data.elasticsearch.annotations.Field.class).type();
} else if (f.isAnnotationPresent(org.springframework.data.elasticsearch.annotations.MultiField.class)) {
type = f.getAnnotation(org.springframework.data.elasticsearch.annotations.MultiField.class).mainField().type();
}
} else {
//如果字段Field type定义的是Keyword,走matchQuery效果也是term精确查询
type = FieldType.Text;
}
if (Objects.equals(type, FieldType.Text)) {
for (Object o : vList) {
if (o instanceof EntityEs.RangeRelation) {
EntityEs.RangeRelation v = (EntityEs.RangeRelation) o;
//范围查询
getRangeBuilder(boolQueryBuilder, k, v);
} else if (o instanceof EntityEs.QueryRelation) {
EntityEs.QueryRelation v = (EntityEs.QueryRelation) o;
if (Objects.equals(v.getQueryMode(), EntityEs.MUST)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.must(QueryBuilders.matchQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.must(QueryBuilders.matchQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.MUST_NOT)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.mustNot(QueryBuilders.matchQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.mustNot(QueryBuilders.matchQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.SHOULD)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.should(QueryBuilders.matchQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.should(QueryBuilders.matchQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.FILTER)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.filter(QueryBuilders.matchQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.filter(QueryBuilders.matchQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
}
}
}
} else if (Objects.equals(type, FieldType.Keyword)) {
for (Object o : vList) {
if (o instanceof EntityEs.RangeRelation) {
EntityEs.RangeRelation v = (EntityEs.RangeRelation) o;
//范围查询
getRangeBuilder(boolQueryBuilder, k, v);
} else if (o instanceof EntityEs.QueryRelation) {
EntityEs.QueryRelation v = (EntityEs.QueryRelation) o;
if (Objects.equals(v.getQueryMode(), EntityEs.MUST)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.must(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.must(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.MUST_NOT)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.mustNot(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.mustNot(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.SHOULD)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.should(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.should(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.FILTER)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.filter(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.filter(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} }
}
} else {
for (Object o : vList) {
if (o instanceof EntityEs.RangeRelation) {
EntityEs.RangeRelation v = (EntityEs.RangeRelation) o;
//范围查询
getRangeBuilder(boolQueryBuilder, k, v);
} else if (o instanceof EntityEs.QueryRelation) {
EntityEs.QueryRelation v = (EntityEs.QueryRelation) o;
if (Objects.equals(v.getQueryMode(), EntityEs.MUST)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.must(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.must(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.MUST_NOT)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.mustNot(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.mustNot(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.SHOULD)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.should(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.should(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.FILTER)) {
if (Objects.isNull(v.getBoostValue())) {
boolQueryBuilder.filter(QueryBuilders.termQuery(k, v.getFieldValue()));
} else {
boolQueryBuilder.filter(QueryBuilders.termQuery(k, v.getFieldValue()).boost(v.getBoostValue()));
}
}
}
}
}
}
}
} /**
* 构建范围查询
*
* @param boolQueryBuilder
* @param k
* @param v
*/
private void getRangeBuilder(BoolQueryBuilder boolQueryBuilder, String k, EntityEs.RangeRelation v) {
if (Objects.equals(v.getQueryMode(), EntityEs.MUST)) {
if (Objects.equals(v.getFieldMinMode(), EntityEs.GT)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).gt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.GTE)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).gte(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LT)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).lt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LTE)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).lte(v.getFieldMinValue()));
}
if (Objects.equals(v.getFieldMaxMode(), EntityEs.GT)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).gt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.GTE)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).gte(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LT)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).lt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LTE)) {
boolQueryBuilder.must(QueryBuilders.rangeQuery(k).lte(v.getFieldMaxValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.MUST_NOT)) {
if (Objects.equals(v.getFieldMinMode(), EntityEs.GT)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).gt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.GTE)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).gte(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LT)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).lt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LTE)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).lte(v.getFieldMinValue()));
}
if (Objects.equals(v.getFieldMaxMode(), EntityEs.GT)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).gt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.GTE)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).gte(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LT)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).lt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LTE)) {
boolQueryBuilder.mustNot(QueryBuilders.rangeQuery(k).lte(v.getFieldMaxValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.SHOULD)) {
if (Objects.equals(v.getFieldMinMode(), EntityEs.GT)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).gt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.GTE)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).gte(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LT)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).lt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LTE)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).lte(v.getFieldMinValue()));
}
if (Objects.equals(v.getFieldMaxMode(), EntityEs.GT)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).gt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.GTE)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).gte(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LT)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).lt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LTE)) {
boolQueryBuilder.should(QueryBuilders.rangeQuery(k).lte(v.getFieldMaxValue()));
}
} else if (Objects.equals(v.getQueryMode(), EntityEs.FILTER)) {
if (Objects.equals(v.getFieldMinMode(), EntityEs.GT)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).gt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.GTE)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).gte(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LT)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).lt(v.getFieldMinValue()));
} else if (Objects.equals(v.getFieldMinMode(), EntityEs.LTE)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).lte(v.getFieldMinValue()));
}
if (Objects.equals(v.getFieldMaxMode(), EntityEs.GT)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).gt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.GTE)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).gte(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LT)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).lt(v.getFieldMaxValue()));
} else if (Objects.equals(v.getFieldMaxMode(), EntityEs.LTE)) {
boolQueryBuilder.filter(QueryBuilders.rangeQuery(k).lte(v.getFieldMaxValue()));
}
} } /**
* 构建排序查询
*
* @param beanIdName
* @param orderMap
* @param nativeSearchQueryBuilder
*/
private void getOrderBuilder(String beanIdName, Map orderMap, NativeSearchQueryBuilder nativeSearchQueryBuilder) {
if (orderMap != null && orderMap.size() > 0) {
orderMap.forEach((k, v) -> {
nativeSearchQueryBuilder.withSort(SortBuilders.fieldSort((String) k).order((SortOrder) v));
});
} else {
//无指定排序字段默认按ID倒序
//nativeSearchQueryBuilder.withSort(SortBuilders.fieldSort(beanIdName).order(SortOrder.DESC));
}
} /**
* 构建高亮查询
*
* @param highlightFields
* @param preTags
* @param postTags
* @param nativeSearchQueryBuilder
*/
private void getHighLightBuilder(List<String> highlightFields, String preTags, String postTags, NativeSearchQueryBuilder nativeSearchQueryBuilder) {
if (highlightFields != null && highlightFields.size() > 0) {
List<HighlightBuilder.Field> highlightTitles = new ArrayList<>();
for (String title : highlightFields) {
highlightTitles.add(new HighlightBuilder.Field(title).preTags(preTags).postTags(postTags));
}
nativeSearchQueryBuilder.withHighlightFields(highlightTitles.stream().toArray(HighlightBuilder.Field[]::new));
}
}
}

五、单元测试

1、根据映射实体设置生成索引与mapping

说明:启动项目会自动加载自动创建,如需要手动创建可调用方法

@Test
public void indexCreateTest() {
System.out.println(esService.indexCreate(EmployeeEs.class));
}

2、删除指定索引

@Test
public void indexDelete() {
System.out.println(esService.indexDelete("employee_index"));
}

3、判断指定索引是否存在

@Test
public void indexExists() {
System.out.println(esService.indexExists(EmployeeEs.class));
}

4、保存单个bean

@Autowired
private EsService esService; @Test
public void saveBean() {
System.out.println("-----es测试start...");
EmployeeEs employee = EmployeeEs.builder().userId(9L).userName("张三1").userCode("abc").userAge(22).userMobile("12345678987")
.remarks("今天天气好晴朗~").birthDay(new Date()).build();
esService.saveBean(employee);
System.out.println("-----es测试end...");
}

5、批量保存

@Test
public void saveList() throws ParseException {
System.out.println("-----es测试start...");
EmployeeEs employee = EmployeeEs.builder().userId(1L).userName("张三").userCode("abc").userAge(18).userSex("男").userMobile("12345678987").birthDay(new SimpleDateFormat("yyyy-MM-dd").parse("1995-9-20")).build();
EmployeeEs employee1 = EmployeeEs.builder().userId(2L).userName("李四").userCode("abc").userAge(10).userSex("女").userMobile("12345678987").birthDay(new SimpleDateFormat("yyyy-MM-dd").parse("1995-6-20")).build();
EmployeeEs employee2 = EmployeeEs.builder().userId(3L).userName("王五").userCode("abc").userAge(10).userSex("男").userMobile("12345678987").birthDay(new SimpleDateFormat("yyyy-MM-dd").parse("1995-5-20")).build();
EmployeeEs employee3 = EmployeeEs.builder().userId(4L).userName("赵六").userCode("abc").userAge(20).userSex("男").userMobile("12345678987").birthDay(new SimpleDateFormat("yyyy-MM-dd").parse("1995-8-20")).build();
EmployeeEs employee4 = EmployeeEs.builder().userId(5L).userName("董七").userCode("abc").userAge(20).userSex("女").userMobile("12345678987").birthDay(new SimpleDateFormat("yyyy-MM-dd").parse("1995-8-20")).build(); esService.saveList(Lists.newArrayList(employee,employee1,employee2,employee3,employee4));
System.out.println("-----es测试end...");
}

6、根据ID删除指定Bean

@Test
public void deleteByBean() {
EmployeeEs employee = EmployeeEs.builder().userId(1L).build();
esService.deleteByBean(employee);
}

7、批量删除

@Test
public void deleteList() {
EmployeeEs employee = EmployeeEs.builder().userId(1L).build();
EmployeeEs employee1 = EmployeeEs.builder().userId(2L).build();
esService.deleteAllByBeanList(Lists.newArrayList(employee,employee1));
}

8、删除该索引下所有数据

@Test
public void deleteAll() {
esService.deleteAll(EmployeeEs.class);
}

9、修改数据(ID不能为空)

@Test
public void updateTest() throws IllegalAccessException {
System.out.println("-----es测试start...");
EmployeeEs employee = EmployeeEs.builder().userId(5L).userName("张一").userCode("abcD").userAge(19).build();
esService.updateByBean(employee);
System.out.println("-----es测试end...");
}

10、根据ID查询指定Bean

@Test
public void queryById() {
EmployeeEs employeeEs = esService.queryBeanById("2", EmployeeEs.class);
System.out.println(employeeEs);
}

11、批量查询

/**
* 涉及到了组合查询bool、权重、范围查找、排序
*/
@Test
public void queryList() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs();
List queryList = Stream.of(employee.new QueryRelation<String>("张三", EntityEs.SHOULD, 5F), employee.new QueryRelation<String>("李四", EntityEs.SHOULD, 20F)).collect(Collectors.toList());
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, queryList).put(EmployeeEs::getUserAge, employee.new RangeRelation(20, EntityEs.GTE, null, null, EntityEs.MUST)));
//排序查询
employee.setOrderMap(new EsMapUtil().put(EmployeeEs::getUserId, SortOrder.DESC));
List<EmployeeEs> employeeEs = esService.queryList(employee);
System.out.println(employeeEs);
System.out.println("-----es测试end...");
}

打印出来的语句:

查询结果:

12、二级属性查询

@Test
public void querySecondList() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs();
employee.setFieldQueryMap(new EsMapUtil().putStr("userName.trueName", employee.new QueryRelation<String>("张三", EntityEs.MUST)));
List<EmployeeEs> employeeEsList = esService.queryList(employee);
System.out.println(employeeEsList);
System.out.println("-----es测试end...");
}

先准备一条测试数据插入

看下上面查询语句的结果:

由于userName的二级属性trueName的类型是keyword,所以是term精确查找

对比:

@Test
public void querySecondList() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs();
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, employee.new QueryRelation<String>("张三", EntityEs.MUST)));
//employee.setFieldQueryMap(new EsMapUtil().putStr("userName.trueName", employee.new QueryRelation<String>("张三", EntityEs.MUST)));
List<EmployeeEs> employeeEsList = esService.queryList(employee);
System.out.println(employeeEsList);
System.out.println("-----es测试end...");
}

13、分页查询

@Test
public void queryPage() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs();
List<EntityEs.QueryRelation> queryList = Stream.of(employee.new QueryRelation<String>("张三", EntityEs.SHOULD), employee.new QueryRelation<String>("李四", EntityEs.SHOULD)).collect(Collectors.toList());
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, queryList));
//分页
employee.setPageNumber(0);
employee.setPageSize(2);
List<EmployeeEs> employeeEs = esService.queryPage(employee);
System.out.println(employeeEs);
System.out.println("-----es测试end...");
}

14、游标分页查询

@Test
public void queryScrollPage() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
String scrollId = "DXF1ZXJ5QW5kRmV0Y2gBAAAAAAAADJEWMVFWMVBnb1ZSZDZsV1k2Y2JjLVlldw==";
EmployeeEs employee = new EmployeeEs();
if (StringUtils.isNotBlank(scrollId)) {
//如果前端有传scrollId
employee.setScrollId(scrollId);
List<EmployeeEs> employeeEs = esService.queryScrollPage(employee);
System.out.println(employeeEs);
} else {
List<EntityEs.QueryRelation> queryList = Stream.of(employee.new QueryRelation<String>("张三", EntityEs.SHOULD, 20F), employee.new QueryRelation<String>("李四", EntityEs.SHOULD)).collect(Collectors.toList());
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, queryList));
//每页页数
employee.setPageSize(4);
List<EmployeeEs> employeeEs = esService.queryScrollPage(employee);
System.out.println(employeeEs);
}
System.out.println("-----es测试end...");
}

15、多层bool套bool查询

@Test
public void queryMuiltiLayer() throws IllegalAccessException, NoSuchFieldException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs(); //多层bool查询
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserSex, employee.new QueryRelation("男", EntityEs.MUST)));
List<EntityEs.QueryRelation> multiLayerUserAgeList = Stream.of(employee.new QueryRelation(10, EntityEs.SHOULD), employee.new QueryRelation(18, EntityEs.SHOULD)).collect(Collectors.toList());
EntityEs.MultiLayerRelation multiLayerRelation = employee.new MultiLayerRelation(EntityEs.MUST, new EsMapUtil().put(EmployeeEs::getUserAge, multiLayerUserAgeList));
employee.setMultiLayerQueryList(Lists.newArrayList(multiLayerRelation)); List<EmployeeEs> userEs = esService.queryList(employee);
System.out.println(userEs);
System.out.println("-----es测试end...");
}

示例:查找年龄为10或者18岁的男性员工

//错误案例
EmployeeEs employee = new EmployeeEs();
List<EntityEs.QueryRelation> ageList = Lists.newArrayList(employee.new QueryRelation(10, EntityEs.SHOULD), employee.new QueryRelation(18, EntityEs.SHOULD));
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserSex, employee.new QueryRelation("男", EntityEs.MUST))
.put(EmployeeEs::getUserAge,ageList)
);
List<EmployeeEs> userEs = esService.queryList(employee);
System.out.println(userEs);

结果年龄为20的赵六也被查询了出来。原因是:当should遇到must和filter时就不是或者了,而是应该的意思。可通过must嵌套一层解决。

修改为多层bool查询

EmployeeEs employee = new EmployeeEs();
//修改为多层bool查询
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserSex, employee.new QueryRelation("男", EntityEs.MUST)));
List<EntityEs.QueryRelation> multiLayerUserAgeList = Stream.of(employee.new QueryRelation(10, EntityEs.SHOULD), employee.new QueryRelation(18, EntityEs.SHOULD)).collect(Collectors.toList());
EntityEs.MultiLayerRelation multiLayerRelation = employee.new MultiLayerRelation(EntityEs.MUST, new EsMapUtil().put(EmployeeEs::getUserAge, multiLayerUserAgeList));
employee.setMultiLayerQueryList(Lists.newArrayList(multiLayerRelation)); List<EmployeeEs> userEs = esService.queryList(employee);

16、高亮查询

@Test
public void highlightTest() throws NoSuchFieldException, IllegalAccessException {
System.out.println("-----es测试start...");
EmployeeEs employee = new EmployeeEs();
employee.setFieldQueryMap(new EsMapUtil().put(EmployeeEs::getUserName, employee.new QueryRelation<String>("张三", EntityEs.MUST))
.put(EmployeeEs::getRemarks, employee.new QueryRelation<String>("天气", EntityEs.MUST)));
employee.setHighlightFields(Lists.newArrayList("remarks"));
//默认<em></em>,可自定义高亮标签
employee.setPreTags("<h1>");
employee.setPostTags("</h1>");
List<EmployeeEs> employeeEs = esService.queryList(employee);
System.out.println(employeeEs);
System.out.println("-----es测试end...");
}

17、聚合查询

@Test
public void queryForAggregation() throws NoSuchFieldException, IllegalAccessException {
System.out.println("-----queryForAggregation-es测试start...");
EmployeeEs employee = new EmployeeEs();
employee.setAggregationMap(new EsMapUtil().put(EmployeeEs::getUserAge, EntityEs.COUNT));
Map employeeEsAggMap = esService.queryForAggregation(employee);
System.out.println("返回结果:" + employeeEsAggMap);
System.out.println("-----queryForAggregation-es测试end...");
}

将上面的EntityEs.COUNT改为EntityEs.SUM 求和运行结果:

elasticsearch通用工具类的更多相关文章

  1. C#编写了一个基于Lucene.Net的搜索引擎查询通用工具类:SearchEngineUtil

    最近由于工作原因,一直忙于公司的各种项目(大部份都是基于spring cloud的微服务项目),故有一段时间没有与大家分享总结最近的技术研究成果的,其实最近我一直在不断的深入研究学习Spring.Sp ...

  2. Android RecyclerView单击、长按事件:基于OnItemTouchListener +GestureDetector标准实现(二),封装抽取成通用工具类

     Android RecyclerView单击.长按事件:基于OnItemTouchListener +GestureDetector标准实现(二),封装抽取成通用工具类 我写的附录文章2,介绍了 ...

  3. java 实现视频转换通用工具类:视频截图-Ffmpeg(四)

    java 实现视频转换通用工具类:获取视频元数据信息(一) java 实现视频转换通用工具类:视频相互转换-总方法及Mencoder(二) java 实现视频转换通用工具类:视频相互转换-Ffmpeg ...

  4. java 实现视频转换通用工具类:视频相互转换-Ffmpeg(三)

    java 实现视频转换通用工具类:获取视频元数据信息(一) java 实现视频转换通用工具类:视频相互转换-总方法及Mencoder(二) 这节主要是ffmpeg的相关方法封装,在实际调用中主要使用f ...

  5. java 实现视频转换通用工具类:视频相互转换-总方法及Mencoder(二)

    1.自动判断格式并调用相应的转换工具,默认方法 /** * 自动判断格式并调用相应的转换工具,默认方法 * @param srcVideoPath * @param tarVideoPath * @r ...

  6. 提取jedis源码的一致性hash代码作为通用工具类

    一致性Hash热点 一致性Hash算法是来解决热点问题,如果虚拟节点设置过小热点问题仍旧存在. 关于一致性Hash算法的原理我就不说了,网上有很多人提供自己编写的一致性Hash算法的代码示例,我在跑网 ...

  7. Java入门:练习——自定义通用工具类

    请编写一个通用工具类,该类具有如下功能: 1)判断一个字符串是否是邮箱地址 2)判断一个字符串是否是手机号码 3)判断一个字符串是否是电话号码 4)判断一个字符串是否是IP地址 代码结构如下,请补充完 ...

  8. Unity3d通用工具类之NGUI图集分解

    ---恢复内容开始--- Unity3d通用工具类之NGUI图集分解 由于最近需要一些美术资源吗,但是无奈自己不会制作UI,所以就打算去网上的项目中直接找几张可以使用的贴图资源. 但是发现这些资源已经 ...

  9. Unity3d通用工具类之数据配置加载类-ini配置文件加载

    Unity3d通用工具类之数据配置加载类-ini配置文件加载 上次我们讲过xml文件的加载配置管理,今天我们换个配置文件,也是比较常见的配置文件.ini格式的数据. 按照国际管理先贴一张啥是.ini文 ...

随机推荐

  1. JavaWeb开发获取客户IP地址

    原创:转载需注明原创地址 https://www.cnblogs.com/fanerwei222/p/11737637.html 本地调试如果使用的是localhost进行访问, 则会获取到 0:0: ...

  2. 如何使Label显示时,一行顶部居中,两行靠左显示----董鑫

    有时我们会碰到这种情况,一个要根据内容显示一行还是两行,一行时还要靠着顶部再居中,比如下面 最左边的名称,要求是靠上的,如果按照正常的方式写的话,可能一行的话就会出现居中显示了,不会顶着头部显示. 我 ...

  3. DNS域名解析之反向解析and主从域名服务器 (今天大小便正常,未来可期)

    DNS解析之反向解析和域名主从服务器 反向解析:根据IP地址查找对应的域名 yum -y install bind 安装软件包 查看需要修改的配置文件所在路径 rpm -qc bind 查询bind软 ...

  4. go基础——基本数据类型

    GO语言的数据类型: /* GO语言的数据类型: 1.基本数据类型: 布尔类型:true,false 数值类型:整数,浮点,复数complex 字符串:string 2.复合数据类型 array,sl ...

  5. VUE动态生成table表格(element-ui)(新增/删除)

    (直接复制即可测试) 结构(红色部分 data/prop/v-model 数据绑定): <template> <el-table size="small" :da ...

  6. 描述nginx中worker_processes、worker_cpu_affinity、worker_rlimit_nofile、worker_connections配置项的含义

    worker_processes worker进程的数量,应小于等于cpu核心数,auto为当前主机cpu核心数 work_processes 4 worker_cpu_affinity 配置CPU亲 ...

  7. 《PHP程序员面试笔试宝典》——如果面试问题曾经遇见过,是否要告知面试官?

    如何巧妙地回答面试官的问题? 本文摘自<PHP程序员面试笔试宝典> 面试中,大多数题目都不是凭空想象出来的,而是有章可循,只要求职者肯花时间,耐得住寂寞,复习得当,基本上在面试前都会见过相 ...

  8. Solution -「ZJOI2012」「洛谷 P2597」灾难

    \(\mathcal{Description}\)   link.   给定一个捕食网络,对于每个物种,求其灭绝后有多少消费者失去所有食物来源.(一些名词与生物学的定义相同 w.)   原图结点数 \ ...

  9. 【Azure 应用服务】部署Jar到App Service for Linux,因启动命令路径配置错误而引起:( Application Error 问题

    问题描述 App Service for Linux 资源创建完成后,通过FTP方式把 .jar包(logdemo.jar)包上传到 /site/wwwroot/ 文件夹后,在App Service的 ...

  10. Vue 源码解读(2)—— Vue 初始化过程

    当学习成为了习惯,知识也就变成了常识. 感谢各位的 点赞.收藏和评论. 新视频和文章会第一时间在微信公众号发送,欢迎关注:李永宁lyn 文章已收录到 github 仓库 liyongning/blog ...