Elasticsearch提供的Java客户端有一些不太方便的地方:

  • 很多地方需要拼接Json字符串,在java中拼接字符串有多恐怖你应该懂的
  • 需要自己把对象序列化为json存储
  • 查询到结果也需要自己反序列化为对象

因此,这里就不讲解原生的Elasticsearch客户端API了。

而是学习Spring提供的套件:Spring Data Elasticsearch。

Spring Data Elasticsearch是Spring Data项目下的一个子模块。

查看 Spring Data的官网:http://projects.spring.io/spring-data/

Spring Data的使命是为数据访问提供熟悉且一致的基于Spring的编程模型,同时仍保留底层数据存储的特殊特性。

它使得使用数据访问技术,关系数据库和非关系数据库,map-reduce框架和基于云的数据服务变得容易。这是一个总括项目,其中包含许多特定于给定数据库的子项目。这些令人兴奋的技术项目背后,是由许多公司和开发人员合作开发的。

Spring Data 的使命是给各种数据访问提供统一的编程接口,不管是关系型数据库(如MySQL),还是非关系数据库(如Redis),或者类似Elasticsearch这样的索引数据库。从而简化开发人员的代码,提高开发效率。

包含很多不同数据操作的模块:

Spring Data Elasticsearch的页面:https://projects.spring.io/spring-data-elasticsearch/

特征:

  • 支持Spring的基于@Configuration的java配置方式,或者XML配置方式
  • 提供了用于操作ES的便捷工具类ElasticsearchTemplate。包括实现文档到POJO之间的自动智能映射。
  • 利用Spring的数据转换服务实现的功能丰富的对象映射
  • 基于注解的元数据映射方式,而且可扩展以支持更多不同的数据格式
  • 根据持久层接口自动生成对应实现方法,无需人工编写基本操作代码(类似mybatis,根据接口自动得到实现)。当然,也支持人工定制查询

我们新建一个demo,学习Elasticsearch

pom依赖:

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
  3. xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
  4. <modelVersion>4.0.0</modelVersion>
  5. <groupId>com.leyou.demo</groupId>
  6. <artifactId>elasticsearch</artifactId>
  7. <version>0.0.1-SNAPSHOT</version>
  8. <packaging>jar</packaging>
  9. <name>elasticsearch</name>
  10. <description>Demo project for Spring Boot</description>
  11. <parent>
  12. <groupId>org.springframework.boot</groupId>
  13. <artifactId>spring-boot-starter-parent</artifactId>
  14. <version>2.0.2.RELEASE</version>
  15. <relativePath/> <!-- lookup parent from repository -->
  16. </parent>
  17. <properties>
  18. <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding>
  19. <project.reporting.outputEncoding>UTF-8</project.reporting.outputEncoding>
  20. <java.version>1.8</java.version>
  21. </properties>
  22. <dependencies>
  23. <dependency>
  24. <groupId>org.springframework.boot</groupId>
  25. <artifactId>spring-boot-starter-data-elasticsearch</artifactId>
  26. </dependency>
  27. <dependency>
  28. <groupId>org.springframework.boot</groupId>
  29. <artifactId>spring-boot-starter-test</artifactId>
  30. <scope>test</scope>
  31. </dependency>
  32. </dependencies>
  33. <build>
  34. <plugins>
  35. <plugin>
  36. <groupId>org.springframework.boot</groupId>
  37. <artifactId>spring-boot-maven-plugin</artifactId>
  38. </plugin>
  39. </plugins>
  40. </build>
  41. </project>

application.yml文件配置:

  1. spring:
  2. data:
  3. elasticsearch:
  4. cluster-name: elasticsearch
  5. cluster-nodes: 192.168.56.101:9300

首先我们准备好实体类:

  1. public class Item {
  2. Long id;
  3. String title; //标题
  4. String category;// 分类
  5. String brand; // 品牌
  6. Double price; // 价格
  7. String images; // 图片地址
  8. }

映射

Spring Data通过注解来声明字段的映射属性,有下面的三个注解:

  • @Document 作用在类,标记实体类为文档对象,一般有两个属性

    • indexName:对应索引库名称
    • type:对应在索引库中的类型
    • shards:分片数量,默认5
    • replicas:副本数量,默认1
  • @Id 作用在成员变量,标记一个字段作为id主键
  • @Field 作用在成员变量,标记为文档的字段,并指定字段映射属性:

    • type:字段类型,取值是枚举:FieldType
    • index:是否索引,布尔类型,默认是true
    • store:是否存储,布尔类型,默认是false
    • analyzer:分词器名称

示例:

  1. @Document(indexName = "item",type = "docs", shards = 1, replicas = 0)
  2. public class Item {
  3. @Id
  4. private Long id;
  5. @Field(type = FieldType.Text, analyzer = "ik_max_word")
  6. private String title; //标题
  7. @Field(type = FieldType.Keyword)
  8. private String category;// 分类
  9. @Field(type = FieldType.Keyword)
  10. private String brand; // 品牌
  11. @Field(type = FieldType.Double)
  12. private Double price; // 价格
  13. @Field(index = false, type = FieldType.Keyword)
  14. private String images; // 图片地址
  15. }

创建索引

ElasticsearchTemplate中提供了创建索引的API:

可以根据类的信息自动生成,也可以手动指定indexName和Settings

映射

映射相关的API:

可以根据类的字节码信息(注解配置)来生成映射,或者手动编写映射

我们这里采用类的字节码信息创建索引并映射:

  1. @RunWith(SpringRunner.class)
  2. @SpringBootTest(classes = ItcastElasticsearchApplication.class)
  3. public class IndexTest {
  4. @Autowired
  5. private ElasticsearchTemplate elasticsearchTemplate;
  6. @Test
  7. public void testCreate(){
  8. // 创建索引,会根据Item类的@Document注解信息来创建
  9. elasticsearchTemplate.createIndex(Item.class);
  10. // 配置映射,会根据Item类中的id、Field等字段来自动完成映射
  11. elasticsearchTemplate.putMapping(Item.class);
  12. }
  13. }

结果:

  1. GET /item
  2. {
  3. "item": {
  4. "aliases": {},
  5. "mappings": {
  6. "docs": {
  7. "properties": {
  8. "brand": {
  9. "type": "keyword"
  10. },
  11. "category": {
  12. "type": "keyword"
  13. },
  14. "images": {
  15. "type": "keyword",
  16. "index": false
  17. },
  18. "price": {
  19. "type": "double"
  20. },
  21. "title": {
  22. "type": "text",
  23. "analyzer": "ik_max_word"
  24. }
  25. }
  26. }
  27. },
  28. "settings": {
  29. "index": {
  30. "refresh_interval": "1s",
  31. "number_of_shards": "1",
  32. "provided_name": "item",
  33. "creation_date": "1525405022589",
  34. "store": {
  35. "type": "fs"
  36. },
  37. "number_of_replicas": "0",
  38. "uuid": "4sE9SAw3Sqq1aAPz5F6OEg",
  39. "version": {
  40. "created": "6020499"
  41. }
  42. }
  43. }
  44. }
  45. }

删除索引的API:

可以根据类名或索引名删除。

示例:

  1. @Test
  2. public void deleteIndex() {
  3. esTemplate.deleteIndex("heima");
  4. }

结果:

Spring Data 的强大之处,就在于你不用写任何DAO处理,自动根据方法名或类的信息进行CRUD操作。只要你定义一个接口,然后继承Repository提供的一些子接口,就能具备各种基本的CRUD功能。

我们只需要定义接口,然后继承它就OK了。

  1. public interface ItemRepository extends ElasticsearchRepository<Item,Long> {
  2. }

来看下Repository的继承关系:

我们看到有一个ElasticsearchRepository接口:

  1. @Autowired
  2. private ItemRepository itemRepository;
  3. @Test
  4. public void index() {
  5. Item item = new Item(1L, "小米手机7", " 手机",
  6. "小米", 3499.00, "http://image.leyou.com/13123.jpg");
  7. itemRepository.save(item);
  8. }

去页面查询看看:

  1. GET /item/_search

结果:

  1. {
  2. "took": 14,
  3. "timed_out": false,
  4. "_shards": {
  5. "total": 1,
  6. "successful": 1,
  7. "skipped": 0,
  8. "failed": 0
  9. },
  10. "hits": {
  11. "total": 1,
  12. "max_score": 1,
  13. "hits": [
  14. {
  15. "_index": "item",
  16. "_type": "docs",
  17. "_id": "1",
  18. "_score": 1,
  19. "_source": {
  20. "id": 1,
  21. "title": "小米手机7",
  22. "category": " 手机",
  23. "brand": "小米",
  24. "price": 3499,
  25. "images": "http://image.leyou.com/13123.jpg"
  26. }
  27. }
  28. ]
  29. }
  30. }

代码:

  1. @Test
  2. public void indexList() {
  3. List<Item> list = new ArrayList<>();
  4. list.add(new Item(2L, "坚果手机R1", " 手机", "锤子", 3699.00, "http://image.leyou.com/123.jpg"));
  5. list.add(new Item(3L, "华为META10", " 手机", "华为", 4499.00, "http://image.leyou.com/3.jpg"));
  6. // 接收对象集合,实现批量新增
  7. itemRepository.saveAll(list);
  8. }

再次去页面查询:

  1. {
  2. "took": 5,
  3. "timed_out": false,
  4. "_shards": {
  5. "total": 1,
  6. "successful": 1,
  7. "skipped": 0,
  8. "failed": 0
  9. },
  10. "hits": {
  11. "total": 3,
  12. "max_score": 1,
  13. "hits": [
  14. {
  15. "_index": "item",
  16. "_type": "docs",
  17. "_id": "2",
  18. "_score": 1,
  19. "_source": {
  20. "id": 2,
  21. "title": "坚果手机R1",
  22. "category": " 手机",
  23. "brand": "锤子",
  24. "price": 3699,
  25. "images": "http://image.leyou.com/13123.jpg"
  26. }
  27. },
  28. {
  29. "_index": "item",
  30. "_type": "docs",
  31. "_id": "3",
  32. "_score": 1,
  33. "_source": {
  34. "id": 3,
  35. "title": "华为META10",
  36. "category": " 手机",
  37. "brand": "华为",
  38. "price": 4499,
  39. "images": "http://image.leyou.com/13123.jpg"
  40. }
  41. },
  42. {
  43. "_index": "item",
  44. "_type": "docs",
  45. "_id": "1",
  46. "_score": 1,
  47. "_source": {
  48. "id": 1,
  49. "title": "小米手机7",
  50. "category": " 手机",
  51. "brand": "小米",
  52. "price": 3499,
  53. "images": "http://image.leyou.com/13123.jpg"
  54. }
  55. }
  56. ]
  57. }
  58. }

修改和新增是同一个接口,区分的依据就是id,这一点跟我们在页面发起PUT请求是类似的。

ElasticsearchRepository提供了一些基本的查询方法:

我们来试试查询所有:

  1. @Test
  2. public void testFind(){
  3. // 查询全部,并安装价格降序排序
  4. Iterable<Item> items = this.itemRepository.findAll(Sort.by(Sort.Direction.DESC, "price"));
  5. items.forEach(item-> System.out.println(item));
  6. }

结果:

Spring Data 的另一个强大功能,是根据方法名称自动实现功能。

比如:你的方法名叫做:findByTitle,那么它就知道你是根据title查询,然后自动帮你完成,无需写实现类。

当然,方法名称要符合一定的约定:

Keyword Sample Elasticsearch Query String
And findByNameAndPrice {"bool" : {"must" : [ {"field" : {"name" : "?"}}, {"field" : {"price" : "?"}} ]}}
Or findByNameOrPrice {"bool" : {"should" : [ {"field" : {"name" : "?"}}, {"field" : {"price" : "?"}} ]}}
Is findByName {"bool" : {"must" : {"field" : {"name" : "?"}}}}
Not findByNameNot {"bool" : {"must_not" : {"field" : {"name" : "?"}}}}
Between findByPriceBetween {"bool" : {"must" : {"range" : {"price" : {"from" : ?,"to" : ?,"include_lower" : true,"include_upper" : true}}}}}
LessThanEqual findByPriceLessThan {"bool" : {"must" : {"range" : {"price" : {"from" : null,"to" : ?,"include_lower" : true,"include_upper" : true}}}}}
GreaterThanEqual findByPriceGreaterThan {"bool" : {"must" : {"range" : {"price" : {"from" : ?,"to" : null,"include_lower" : true,"include_upper" : true}}}}}
Before findByPriceBefore {"bool" : {"must" : {"range" : {"price" : {"from" : null,"to" : ?,"include_lower" : true,"include_upper" : true}}}}}
After findByPriceAfter {"bool" : {"must" : {"range" : {"price" : {"from" : ?,"to" : null,"include_lower" : true,"include_upper" : true}}}}}
Like findByNameLike {"bool" : {"must" : {"field" : {"name" : {"query" : "?*","analyze_wildcard" : true}}}}}
StartingWith findByNameStartingWith {"bool" : {"must" : {"field" : {"name" : {"query" : "?*","analyze_wildcard" : true}}}}}
EndingWith findByNameEndingWith {"bool" : {"must" : {"field" : {"name" : {"query" : "*?","analyze_wildcard" : true}}}}}
Contains/Containing findByNameContaining {"bool" : {"must" : {"field" : {"name" : {"query" : "**?**","analyze_wildcard" : true}}}}}
In findByNameIn(Collection<String>names) {"bool" : {"must" : {"bool" : {"should" : [ {"field" : {"name" : "?"}}, {"field" : {"name" : "?"}} ]}}}}
NotIn findByNameNotIn(Collection<String>names) {"bool" : {"must_not" : {"bool" : {"should" : {"field" : {"name" : "?"}}}}}}
Near findByStoreNear Not Supported Yet !
True findByAvailableTrue {"bool" : {"must" : {"field" : {"available" : true}}}}
False findByAvailableFalse {"bool" : {"must" : {"field" : {"available" : false}}}}
OrderBy findByAvailableTrueOrderByNameDesc {"sort" : [{ "name" : {"order" : "desc"} }],"bool" : {"must" : {"field" : {"available" : true}}}}

例如,我们来按照价格区间查询,定义这样的一个方法:

  1. public interface ItemRepository extends ElasticsearchRepository<Item,Long> {
  2. /**
  3. * 根据价格区间查询
  4. * @param price1
  5. * @param price2
  6. * @return
  7. */
  8. List<Item> findByPriceBetween(double price1, double price2);
  9. }

然后添加一些测试数据:

  1. @Test
  2. public void indexList() {
  3. List<Item> list = new ArrayList<>();
  4. list.add(new Item(1L, "小米手机7", "手机", "小米", 3299.00, "http://image.leyou.com/13123.jpg"));
  5. list.add(new Item(2L, "坚果手机R1", "手机", "锤子", 3699.00, "http://image.leyou.com/13123.jpg"));
  6. list.add(new Item(3L, "华为META10", "手机", "华为", 4499.00, "http://image.leyou.com/13123.jpg"));
  7. list.add(new Item(4L, "小米Mix2S", "手机", "小米", 4299.00, "http://image.leyou.com/13123.jpg"));
  8. list.add(new Item(5L, "荣耀V10", "手机", "华为", 2799.00, "http://image.leyou.com/13123.jpg"));
  9. // 接收对象集合,实现批量新增
  10. itemRepository.saveAll(list);
  11. }

不需要写实现类,然后我们直接去运行:

  1. @Test
  2. public void queryByPriceBetween(){
  3. List<Item> list = this.itemRepository.findByPriceBetween(2000.00, 3500.00);
  4. for (Item item : list) {
  5. System.out.println("item = " + item);
  6. }
  7. }

结果:

虽然基本查询和自定义方法已经很强大了,但是如果是复杂查询(模糊、通配符、词条查询等)就显得力不从心了。此时,我们只能使用原生查询。

先看看基本玩法

  1. @Test
  2. public void testQuery(){
  3. // 词条查询
  4. MatchQueryBuilder queryBuilder = QueryBuilders.matchQuery("title", "小米");
  5. // 执行查询
  6. Iterable<Item> items = this.itemRepository.search(queryBuilder);
  7. items.forEach(System.out::println);
  8. }

Repository的search方法需要QueryBuilder参数,elasticSearch为我们提供了一个对象QueryBuilders:

QueryBuilders提供了大量的静态方法,用于生成各种不同类型的查询对象,例如:词条、模糊、通配符等QueryBuilder对象。

结果:

elasticsearch提供很多可用的查询方式,但是不够灵活。如果想玩过滤或者聚合查询等就很难了。

先来看最基本的match query:

  1. @Test
  2. public void testNativeQuery(){
  3. // 构建查询条件
  4. NativeSearchQueryBuilder queryBuilder = new NativeSearchQueryBuilder();
  5. // 添加基本的分词查询
  6. queryBuilder.withQuery(QueryBuilders.matchQuery("title", "小米"));
  7. // 执行搜索,获取结果
  8. Page<Item> items = this.itemRepository.search(queryBuilder.build());
  9. // 打印总条数
  10. System.out.println(items.getTotalElements());
  11. // 打印总页数
  12. System.out.println(items.getTotalPages());
  13. items.forEach(System.out::println);
  14. }

NativeSearchQueryBuilder:Spring提供的一个查询条件构建器,帮助构建json格式的请求体

Page<item>:默认是分页查询,因此返回的是一个分页的结果对象,包含属性:

  • totalElements:总条数
  • totalPages:总页数
  • Iterator:迭代器,本身实现了Iterator接口,因此可直接迭代得到当前页的数据
  • 其它属性:

结果:

利用NativeSearchQueryBuilder可以方便的实现分页:

  1. @Test
  2. public void testNativeQuery(){
  3. // 构建查询条件
  4. NativeSearchQueryBuilder queryBuilder = new NativeSearchQueryBuilder();
  5. // 添加基本的分词查询
  6. queryBuilder.withQuery(QueryBuilders.termQuery("category", "手机"));
  7. // 初始化分页参数
  8. int page = 0;
  9. int size = 3;
  10. // 设置分页参数
  11. queryBuilder.withPageable(PageRequest.of(page, size));
  12. // 执行搜索,获取结果
  13. Page<Item> items = this.itemRepository.search(queryBuilder.build());
  14. // 打印总条数
  15. System.out.println(items.getTotalElements());
  16. // 打印总页数
  17. System.out.println(items.getTotalPages());
  18. // 每页大小
  19. System.out.println(items.getSize());
  20. // 当前页
  21. System.out.println(items.getNumber());
  22. items.forEach(System.out::println);
  23. }

结果:

可以发现,Elasticsearch中的分页是从第0页开始

排序也通用通过NativeSearchQueryBuilder完成:

  1. @Test
  2. public void testSort(){
  3. // 构建查询条件
  4. NativeSearchQueryBuilder queryBuilder = new NativeSearchQueryBuilder();
  5. // 添加基本的分词查询
  6. queryBuilder.withQuery(QueryBuilders.termQuery("category", "手机"));
  7. // 排序
  8. queryBuilder.withSort(SortBuilders.fieldSort("price").order(SortOrder.DESC));
  9. // 执行搜索,获取结果
  10. Page<Item> items = this.itemRepository.search(queryBuilder.build());
  11. // 打印总条数
  12. System.out.println(items.getTotalElements());
  13. items.forEach(System.out::println);
  14. }

结果:

桶就是分组,比如这里我们按照品牌brand进行分组:

  1. @Test
  2. public void testAgg(){
  3. NativeSearchQueryBuilder queryBuilder = new NativeSearchQueryBuilder();
  4. // 不查询任何结果
  5. queryBuilder.withSourceFilter(new FetchSourceFilter(new String[]{""}, null));
  6. // 1、添加一个新的聚合,聚合类型为terms,聚合名称为brands,聚合字段为brand
  7. queryBuilder.addAggregation(
  8. AggregationBuilders.terms("brands").field("brand"));
  9. // 2、查询,需要把结果强转为AggregatedPage类型
  10. AggregatedPage<Item> aggPage = (AggregatedPage<Item>) this.itemRepository.search(queryBuilder.build());
  11. // 3、解析
  12. // 3.1、从结果中取出名为brands的那个聚合,
  13. // 因为是利用String类型字段来进行的term聚合,所以结果要强转为StringTerm类型
  14. StringTerms agg = (StringTerms) aggPage.getAggregation("brands");
  15. // 3.2、获取桶
  16. List<StringTerms.Bucket> buckets = agg.getBuckets();
  17. // 3.3、遍历
  18. for (StringTerms.Bucket bucket : buckets) {
  19. // 3.4、获取桶中的key,即品牌名称
  20. System.out.println(bucket.getKeyAsString());
  21. // 3.5、获取桶中的文档数量
  22. System.out.println(bucket.getDocCount());
  23. }
  24. }

显示的结果:

关键API:

  • AggregationBuilders:聚合的构建工厂类。所有聚合都由这个类来构建,看看他的静态方法:

  • AggregatedPage:聚合查询的结果类。它是Page<T>的子接口:

AggregatedPagePage功能的基础上,拓展了与聚合相关的功能,它其实就是对聚合结果的一种封装,大家可以对照聚合结果的JSON结构来看。

而返回的结果都是Aggregation类型对象,不过根据字段类型不同,又有不同的子类表示

我们看下页面的查询的JSON结果与Java类的对照关系:

代码:

  1. @Test
  2. public void testSubAgg(){
  3. NativeSearchQueryBuilder queryBuilder = new NativeSearchQueryBuilder();
  4. // 不查询任何结果
  5. queryBuilder.withSourceFilter(new FetchSourceFilter(new String[]{""}, null));
  6. // 1、添加一个新的聚合,聚合类型为terms,聚合名称为brands,聚合字段为brand
  7. queryBuilder.addAggregation(
  8. AggregationBuilders.terms("brands").field("brand")
  9. .subAggregation(AggregationBuilders.avg("priceAvg").field("price")) // 在品牌聚合桶内进行嵌套聚合,求平均值
  10. );
  11. // 2、查询,需要把结果强转为AggregatedPage类型
  12. AggregatedPage<Item> aggPage = (AggregatedPage<Item>) this.itemRepository.search(queryBuilder.build());
  13. // 3、解析
  14. // 3.1、从结果中取出名为brands的那个聚合,
  15. // 因为是利用String类型字段来进行的term聚合,所以结果要强转为StringTerm类型
  16. StringTerms agg = (StringTerms) aggPage.getAggregation("brands");
  17. // 3.2、获取桶
  18. List<StringTerms.Bucket> buckets = agg.getBuckets();
  19. // 3.3、遍历
  20. for (StringTerms.Bucket bucket : buckets) {
  21. // 3.4、获取桶中的key,即品牌名称 3.5、获取桶中的文档数量
  22. System.out.println(bucket.getKeyAsString() + ",共" + bucket.getDocCount() + "台");
  23. // 3.6.获取子聚合结果:
  24. InternalAvg avg = (InternalAvg) bucket.getAggregations().asMap().get("priceAvg");
  25. System.out.println("平均售价:" + avg.getValue());
  26. }
  27. }

结果:

版权声明:本文为jimlau原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://www.cnblogs.com/jimlau/p/12162128.html