广告
返回顶部
首页 > 资讯 > 数据库 >为什么查询ElasticSearch用SQL代替DSL
  • 147
分享到

为什么查询ElasticSearch用SQL代替DSL

2024-04-02 19:04:59 147人浏览 八月长安
摘要

这篇文章主要讲解了“为什么查询elasticsearch用sql代替DSL”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“为什么查询ElasticSearch

这篇文章主要讲解了“为什么查询elasticsearchsql代替DSL”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“为什么查询ElasticSearch用SQL代替DSL”吧!

SQL REST API

在Kibana Console中输入:

POST /_sql?fORMat=txt {   "query": "SELECT * FROM library ORDER BY page_count DESC LIMIT 5" }

将上述SQL替换为你自己的SQL语句,即可。返回格式如下:

    author      |        name        |  page_count   | release_date -----------------+--------------------+---------------+------------------------ Peter F. Hamilton|Pandora's Star      |768            |2004-03-02T00:00:00.000Z Vernor Vinge     |A Fire Upon the Deep|613            |1992-06-01T00:00:00.000Z Frank Herbert    |Dune                |604            |1965-06-01T00:00:00.000Z

SQL CLI

elasticsearch-sql-cli是安装ES时bin目录的一个脚本文件,也可单独下载。我们在ES目录运行

./bin/elasticsearch-sql-cli https://some.server:9200

输入sql即可查询

sql> SELECT * FROM library WHERE page_count > 500 ORDER BY page_count DESC;      author      |        name        |  page_count   | release_date -----------------+--------------------+---------------+--------------- Peter F. Hamilton|Pandora's Star      |768            |1078185600000 Vernor Vinge     |A Fire Upon the Deep|613            |707356800000 Frank Herbert    |Dune                |604            |-144720000000

SQL To DSL

在Kibana输入:

POST /_sql/translate {   "query": "SELECT * FROM library ORDER BY page_count DESC",   "fetch_size": 10 }

即可得到转化后的DSL query:

{   "size": 10,   "docvalue_fields": [     {       "field": "release_date",       "format": "epoch_millis"     }   ],   "_source": {     "includes": [       "author",       "name",       "page_count"     ],     "excludes": []   },   "sort": [     {       "page_count": {         "order": "desc",         "missing": "_first",         "unmapped_type": "short"       }     }   ] }

因为查询相关的语句已经生成,我们只需要在这个基础上适当修改或不修改就可以愉快使用DSL了。

下面我们详细介绍下ES SQL 支持的SQL语句 和 如何避免错误使用。

首先需要了解下ES SQL支持的SQL语句中,SQL术语和ES术语的对应关系:

为什么查询ElasticSearch用SQL代替DSL

ES SQL的语法支持大多遵循ANSI SQL标准,支持的SQL语句有DML查询和部分DDL查询。

DDL查询如:DESCRIBE table,SHOW COLUMNS IN  table略显鸡肋,我们主要看下对SELECT,Function的DML查询支持。

SELECT

语法结构如下:

SELECT [TOP [ count ] ] select_expr [, ...] [ FROM table_name ] [ WHERE condition ] [ GROUP BY grouping_element [, ...] ] [ HAVING condition] [ ORDER BY expression [ ASC | DESC ] [, ...] ] [ LIMIT [ count ] ] [ PIVOT ( aggregation_expr FOR column IN ( value [ [ AS ] alias ] [, ...] ) ) ]

表示从0-N个表中获取行数据。SQL的执行顺序为:

  1. 鸿蒙官方战略合作共建——HarmonyOS技术社区

  2. 获取所有 FROM中的关键词,确定表名。

  3. 如果有WHERE条件,过滤掉所有不符合的行。

  4. 如果有GROUP BY条件,则分组聚合;如果有HAVING条件,则过滤聚合的结果。

  5. 上一步得到的结果经过select_expr运算,确定具体返回的数据。

  6. 如果有 ORDER BY条件,会对返回的数据排序

  7. 如果有 LIMIT or TOP条件,会返回上一步结果的子集。

与常用的SQL有两点不同,ES SQL 支持TOP [ count ]和PIVOT ( aggregation_expr FOR column IN (  value [ [ AS ] alias ] [, ...] ) )子句。

TOP [ count ] :如SELECT TOP 2 first_name FROM emp表示最多返回两条数据,不可与LIMIT条件共用。

PIVOT子句会对其聚合条件得到的结果进行行转列,进一步运算。这个我是没用过,不做介绍。

FUNCTION

基于上面的SQL我们其实已经能有过滤,聚合,排序,分页功能的SQL了。但是我们需要进一步了解ES  SQL中FUNCTION的支持,才能写出丰富的具有全文搜索,聚合,分组功能的SQL。

使用SHOW FUNCTIONS 可列举出支持的函数名称和所属类型。

SHOW FUNCTIONS;        name       |     type -----------------+--------------- AVG              |AGGREGATE COUNT            |AGGREGATE FIRST            |AGGREGATE FIRST_VALUE      |AGGREGATE LAST             |AGGREGATE LAST_VALUE       |AGGREGATE MAX              |AGGREGATE MIN              |AGGREGATE SUM              |AGGREGATE ........

我们主要看下聚合,分组,全文搜索相关的常用函数。

全文匹配函数

MATCH:相当于DSL中的match and multi_match查询。

MATCH(     field_exp,       --字段名称     constant_exp,       --字段的匹配值     [, options])       --可选项

使用举例:

SELECT author, name FROM library WHERE MATCH(author, 'frank');      author     |       name ---------------+------------------- Frank Herbert  |Dune Frank Herbert  |Dune Messiah SELECT author, name, SCORE() FROM library WHERE MATCH('author^2,name^5', 'frank dune');      author     |       name        |    SCORE() ---------------+-------------------+--------------- Frank Herbert  |Dune               |11.443176 Frank Herbert  |Dune Messiah       |9.446629

QUERY:相当于DSL中的 query_string 查询。

QUERY(     constant_exp      --匹配值表达式     [, options])       --可选项

使用举例:

SELECT author, name, page_count, SCORE() FROM library WHERE QUERY('_exists_:"author" AND page_count:>200 AND (name:/star.*/ OR name:duna~)');        author      |       name        |  page_count   |    SCORE() ------------------+-------------------+---------------+--------------- Frank Herbert     |Dune               |604            |3.7164764 Frank Herbert     |Dune Messiah       |331            |3.4169943

SCORE():返回输入数据和返回数据的相关度relevance.

使用举例:

SELECT SCORE(), * FROM library WHERE MATCH(name, 'dune') ORDER BY SCORE() DESC;      SCORE()    |    author     |       name        |  page_count   |    release_date ---------------+---------------+-------------------+---------------+-------------------- 2.2886353      |Frank Herbert  |Dune               |604            |1965-06-01T00:00:00Z 1.8893257      |Frank Herbert  |Dune Messiah       |331            |1969-10-15T00:00:00Z

聚合函数

AVG(numeric_field) :计算数字类型的字段的平均值。

SELECT AVG(salary) AS avg FROM emp;

COUNT(expression):返回输入数据的总数,包括COUNT()时field_name对应的值为null的数据。

COUNT(ALL field_name):返回输入数据的总数,不包括field_name对应的值为null的数据。

COUNT(DISTINCT field_name):返回输入数据中field_name对应的值不为null的总数。

SUM(field_name):返回输入数据中数字字段field_name对应的值的总和。

MIN(field_name):返回输入数据中数字字段field_name对应的值的最小值。

MAX(field_name):返回输入数据中数字字段field_name对应的值的最大值。

分组函数

这里的分组函数是对应DSL中的bucket分组。

HISTOGRAM:语法如下:

HISTOGRAM(            numeric_exp,    --数字表达式,通常是一个field_name            numeric_interval    --数字的区间值 )  HISTOGRAM(            date_exp,      --date/time表达式,通常是一个field_name            date_time_interval      --date/time的区间值 )

如下返回每年1月1号凌晨出生的数据:

ELECT HISTOGRAM(birth_date, INTERVAL 1 YEAR) AS h, COUNT(*) AS c FROM emp GROUP BY h;              h            |       c ------------------------+--------------- null                    |10 1952-01-01T00:00:00.000Z|8 1953-01-01T00:00:00.000Z|11 1954-01-01T00:00:00.000Z|8 1955-01-01T00:00:00.000Z|4 1956-01-01T00:00:00.000Z|5 1957-01-01T00:00:00.000Z|4 1958-01-01T00:00:00.000Z|7 1959-01-01T00:00:00.000Z|9 1960-01-01T00:00:00.000Z|8 1961-01-01T00:00:00.000Z|8 1962-01-01T00:00:00.000Z|6 1963-01-01T00:00:00.000Z|7 1964-01-01T00:00:00.000Z|4 1965-01-01T00:00:00.000Z|1

ES SQL局限性

因为ES SQL和ES DSL在功能上并非完全匹配,官方文档提到的SQL局限性有:

大的查询可能抛ParsingException

在解析阶段,极大的查询会占用过多的内存,在这种情况下,Elasticsearch SQL引擎将中止解析并抛出错误。

nested类型字段的表示方法

SQL中不支持nested类型的字段,只能使用

[nested_field_name].[sub_field_name]

这种形式来引用内嵌子字段。

使用举例:

SELECT dep.dep_name.keyWord FROM test_emp GROUP BY languages;

nested类型字段不能用在where 和 order by 的Scalar函数上

如以下SQL都是错误的

SELECT * FROM test_emp WHERE LENGTH(dep.dep_name.keyword) > 5;  SELECT * FROM test_emp ORDER BY YEAR(dep.start_date);

不支持多个nested字段的同时查询

如嵌套字段nested_A和nested_B无法同时使用。

nested内层字段分页限制

当分页查询有nested字段时,分页结果可能不正确。这是因为:ES中的分页查询发生在Root nested  document上,而不是它的内层字段上。

keyword类型的字段不支持normalizer

不支持数组类型的字段

这是因为在SQL中一个field只对应一个值,这种情况下我们可以使用上面介绍的 SQL To DSL的api  转化为DSL语句,用DSL查询就好了。

聚合排序的限制

  • 排序字段必须是聚合桶中的字段,ES SQL  CLI突破了这种限制,但上限不能超过512行,否则在sorting阶段会抛异常。推荐搭配Limit子句使用,如:

SELECT * FROM test GROUP BY age ORDER BY COUNT(*) LIMIT 100;

聚合排序的排序条件不支持Scalar函数或者简单的操作符运算。聚合后的复杂字段(比如包含聚合函数)也是不能用在排序条件上的。

以下是错误例子:

SELECT age, ROUND(AVG(salary)) AS avg FROM test GROUP BY age ORDER BY avg;  SELECT age, MAX(salary) - MIN(salary) AS diff FROM test GROUP BY age ORDER BY diff;

子查询的限制

子查询中包含GROUP BY or HAVING 或者比SELECT X FROM (SELECT ...) WHERE  [simple_condition]这种结构复杂,都是可能执行不成功的。

TIME 数据类型的字段不支持GROUP BY条件和HISTOGRAM函数

如以下查询是错误的:

SELECT count(*) FROM test GROUP BY CAST(date_created AS TIME);  SELECT HISTOGRAM(CAST(birth_date AS TIME), INTERVAL '10' MINUTES) as h, COUNT(*) FROM t GROUP BY h

但是将TIME类型的字段包装为Scalar函数返回是支持GROUP BY的,如:

SELECT count(*) FROM test GROUP BY MINUTE((CAST(date_created AS TIME));

返回字段的限制

如果一个字段不在source中存储,是无法查询到的。keyword, date, scaled_float, geo_point,  geo_shape这些类型的字段不受这种限制,因为他们不是从_source中返回,而是从docvalue_fields中返回。

感谢各位的阅读,以上就是“为什么查询ElasticSearch用SQL代替DSL”的内容了,经过本文的学习后,相信大家对为什么查询ElasticSearch用SQL代替DSL这一问题有了更深刻的体会,具体使用情况还需要大家实践验证。这里是编程网,小编将为大家推送更多相关知识点的文章,欢迎关注!

您可能感兴趣的文档:

--结束END--

本文标题: 为什么查询ElasticSearch用SQL代替DSL

本文链接: https://www.lsjlt.com/news/62014.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 为什么查询ElasticSearch用SQL代替DSL
    这篇文章主要讲解了“为什么查询ElasticSearch用SQL代替DSL”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“为什么查询ElasticSearch...
    99+
    2022-10-18
  • 怎么用SQL代替DSL
    这篇文章主要为大家展示了“怎么用SQL代替DSL”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“怎么用SQL代替DSL”这篇文章吧。SQL REST API在Kibana Console中输入:P...
    99+
    2023-06-27
  • 参数化查询为什么能避免sql注入
    参数化查询能避免sql注入的原因是采用了预编译方法,先将SQL语句中可被客户端控制的参数集进行编译,生成对应的临时变量集,再使用对应的设置方法,为临时变量集的元素进行赋值,赋值函数为setString(),这样会对传入的参数进行强制类型检查...
    99+
    2022-10-22
  • 【黄啊码】什么是ElasticSearch?它会替代MySQL成为主流吗?如何优化?TP5如何接入ElasticSearch?
    目录 什么是ElasticSearch? ES和MySQL使用场景的比较 Elasticsearch和MySQL具体应用场景: 如何优化: 索引效率优化 批量提交 优化硬件、 减少副本数量 查询效率优化 路由 不带 routing 查询 ...
    99+
    2023-10-06
    elasticsearch 搜索引擎 数据库 php
  • 为什么参数化查询可以防止sql注入
    参数化查询可以防止sql注入原因:参数化查询可以对参数进行过滤,还能够重用执行计划,若执行计划被重用的话,SQL所要表达的语义是不会变化的,因此参数化查询可以防止SQL注入。...
    99+
    2022-10-25
  • 在sql查询中group by有什么用
    小编给大家分享一下在sql查询中group by有什么用,希望大家阅读完这篇文章之后都有所收获,下面让我们一起去探讨吧!在sql查询中group by的含义是用于结合合计函数,根据一个或多个列对结果集进行分...
    99+
    2022-10-18
  • SQL Server解惑——为什么你的查询结果超出了查询时间范围
    废话少说,直接上SQL代码(有兴趣的测试验证一下),下面这个查询语句为什么将2008-11-27的记录查询出来了呢?这个是同事遇到的一个问题,个人设计了一个例子。   USE AdventureWorks2014;GOSELECT...
    99+
    2019-01-06
    SQL Server解惑——为什么你的查询结果超出了查询时间范围
  • 将自然语言查询转换为SQL代码的AI工具怎么使用
    今天小编给大家分享一下将自然语言查询转换为SQL代码的AI工具怎么使用的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。介绍人工...
    99+
    2023-07-05
  • sql server查询分析器的作用是什么
    SQL Server查询分析器是用于分析和优化数据库查询性能的工具。它提供了一个可视化界面,可以帮助开发人员和数据库管理员分析执行计...
    99+
    2023-09-26
    sql server
  • 将自然语言查询转换为SQL代码的AI工具使用详解
    目录介绍功能本地安装使用仓库地址介绍 人工智能可以帮助我们实现将自然语言查询和 SQL 代码进行互相转换,这种技术通过自然语言处理(NLP)和机器学习,训练模型来理解和生成自然语言...
    99+
    2023-03-20
    自然语言查询转换SQL AI工具转换代码
  • SQL查询中什么时候需要使用表别名
    这篇文章主要介绍SQL查询中什么时候需要使用表别名,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!SQL查询中什么时候需要使用表别名?1、 表名很长时select ...
    99+
    2022-10-19
  • 为什么Linux内核常常用Unsigned Long来代替指针
    这篇文章给大家介绍为什么Linux内核常常用Unsigned Long来代替指针,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。昨天我犯了一个错误把指针和整数“混淆”的错误,幸得队友王童鞋指正,今早起床,我把这个心得花一...
    99+
    2023-06-15
  • sql多表内连接查询使用的语句是什么
    这篇文章主要讲解了“sql多表内连接查询使用的语句是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“sql多表内连接查询使用的语句是什么”吧! ...
    99+
    2022-10-18
  • SQL为什么不建议执行超过3表以上的多表关联查询
    目录摘要:MySQL多表关联查询效率高点还是多次单表查询效率高?对关联查询进行分解概述:前段时间在跟其他公司DBA交流时谈到了mysql跟PG之间在多表关联查询上的一些区别,相比之下...
    99+
    2022-11-13
  • 浅谈为什么MySQL不推荐使用子查询和join
    做分页查询: 对于mysql,不推荐使用子查询和join是因为本身join的效率就是硬伤,一旦数据量很大效率就很难保证,强烈推荐分别根据索引单表取数据,然后在程序里面做join,merge数据。 2.子查询就更别用...
    99+
    2022-05-29
    MySQL 子查询和join MySQL 子查询 MySQL join
  • SQL常用日期查询语句及显示格式设置是什么
    本文小编为大家详细介绍“SQL常用日期查询语句及显示格式设置是什么”,内容详细,步骤清晰,细节处理妥当,希望这篇“SQL常用日期查询语句及显示格式设置是什么”文章能帮助大家解决疑惑,下面跟着小编的思路慢慢深入,一起来学习新知识吧。1.查询当...
    99+
    2023-06-29
  • SQL Server中的子查询和表链接概念及使用是什么
    SQL Server中的子查询和表链接概念及使用是什么,很多新手对此不是很清楚,为了帮助大家解决这个难题,下面小编将为大家详细讲解,有这方面需求的人可以来学习下,希望你能有所收获。1.子查询概念 ...
    99+
    2022-10-19
  • oracle数据库中为什么在查询里索引未被使用
    这篇文章主要介绍了oracle数据库中为什么在查询里索引未被使用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。快速检查表上是否存在索引检查您...
    99+
    2022-10-19
  • MySQL SQL性能分析之慢查询日志、explain使用的方法是什么
    本篇内容介绍了“MySQL SQL性能分析之慢查询日志、explain使用的方法是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况...
    99+
    2023-04-19
    mysql sql explain
  • sql查询语句中用于实现选择运算的语法关键词是什么
    这篇文章主要介绍了sql查询语句中用于实现选择运算的语法关键词是什么,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。在sql的selec...
    99+
    2022-10-18
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作