iis服务器助手广告广告
返回顶部
首页 > 资讯 > 精选 >怎么用SQL代替DSL
  • 756
分享到

怎么用SQL代替DSL

2023-06-27 10:06:00 756人浏览 八月长安
摘要

这篇文章主要为大家展示了“怎么用sql代替DSL”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“怎么用SQL代替DSL”这篇文章吧。SQL REST api在Kibana Console中输入:P

这篇文章主要为大家展示了“怎么用sql代替DSL”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“怎么用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.000ZVernor Vinge     |A Fire Upon the Deep|613            |1992-06-01T00:00:00.000ZFrank 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            |1078185600000Vernor Vinge     |A Fire Upon the Deep|613            |707356800000Frank 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术语的对应关系:

怎么用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. 获取所有 FROM中的关键词,确定表名。

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

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

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

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

  6. 如果有 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              |AGGREGATECOUNT            |AGGREGATEFIRST            |AGGREGATEFIRST_VALUE      |AGGREGATELAST             |AGGREGATELAST_VALUE       |AGGREGATEMAX              |AGGREGATEMIN              |AGGREGATESUM              |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  |DuneFrank Herbert  |Dune MessiahSELECT author, name, SCORE() FROM library WHERE MATCH('author^2,name^5', 'frank dune');    author     |       name        |    SCORE()---------------+-------------------+---------------Frank Herbert  |Dune               |11.443176Frank 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.7164764Frank 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:00Z1.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                    |101952-01-01T00:00:00.000Z|81953-01-01T00:00:00.000Z|111954-01-01T00:00:00.000Z|81955-01-01T00:00:00.000Z|41956-01-01T00:00:00.000Z|51957-01-01T00:00:00.000Z|41958-01-01T00:00:00.000Z|71959-01-01T00:00:00.000Z|91960-01-01T00:00:00.000Z|81961-01-01T00:00:00.000Z|81962-01-01T00:00:00.000Z|61963-01-01T00:00:00.000Z|71964-01-01T00:00:00.000Z|41965-01-01T00:00:00.000Z|1

ES SQL局限性

因为ES SQLES 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_Anested_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中返回。

以上是“怎么用SQL代替DSL”这篇文章的所有内容,感谢各位的阅读!相信大家都有了一定的了解,希望分享的内容对大家有所帮助,如果还想学习更多知识,欢迎关注编程网精选频道!

--结束END--

本文标题: 怎么用SQL代替DSL

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

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

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

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

下载Word文档
猜你喜欢
  • 怎么用SQL代替DSL
    这篇文章主要为大家展示了“怎么用SQL代替DSL”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“怎么用SQL代替DSL”这篇文章吧。SQL REST API在Kibana Console中输入:P...
    99+
    2023-06-27
  • 为什么查询ElasticSearch用SQL代替DSL
    这篇文章主要讲解了“为什么查询ElasticSearch用SQL代替DSL”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“为什么查询ElasticSearch...
    99+
    2024-04-02
  • sql中in用什么代替
    sql 中 in 运算符的替代方案包括 exists 子查询、case 表达式和 or 运算符。替代方案的选择取决于性能、灵活性、可维护性等因素,通常 in 运算符是首选,但替代方案在某...
    99+
    2024-05-02
  • sql中or可以用什么代替
    sql 中 or 运算符的替代方案包括:1. union:合并查询结果并丢弃重复记录;2. in:检查值是否包含在指定列表中;3. case when:根据条件返回不同值;4....
    99+
    2024-05-08
  • sql中的in可以用什么代替
    使用 or 运算符可以替代 sql 中的 in 运算符,从而有效率地检查一个值是否包含在指定列表中。or 运算符的优势在于语法直观、易于使用,尤其是在列表值较多时。需要注意的是,or 运...
    99+
    2024-05-02
  • Python中Elasticsearch DSL怎么使用
    这篇文章主要介绍了Python中Elasticsearch DSL怎么使用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。python主要应用领域有哪些1、云计算,典型应用Op...
    99+
    2023-06-14
  • 怎么用mongodb替代es
    要将MongoDB替换为Elasticsearch,您需要采取以下步骤:1. 将MongoDB中的数据导出为JSON格式。您可以使用...
    99+
    2023-09-06
    mongodb
  • sql中的in可以用什么代替数据
    对于 sql 中的 in 替代选项,可考虑以下建议:使用 or 代替较小的值集合;用 case when 创建动态查询,根据条件更改值匹配;利用 join 优化涉及多个表的查询,尤其在值...
    99+
    2024-05-08
  • ALT代替属性怎么用
    这篇文章主要介绍了ALT代替属性怎么用,具有一定借鉴价值,感兴趣的朋友可以参考下,希望大家阅读完这篇文章之后大有收获,下面让小编带着大家一起了解一下。ALT-代替属性,用来对网页上的图片进行描述,光标在图片上时显示的提示语即采用该标签实现,...
    99+
    2023-06-13
  • sql中的替换函数怎么用
    非常抱歉,由于您没有提供文章标题,我无法为您生成一篇高质量的文章。请您提供文章标题,我将尽快为您生成一篇优质的文章。...
    99+
    2024-05-15
  • Linux下ip替代ifconfig怎么用
    小编给大家分享一下Linux下ip替代ifconfig怎么用,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!如果你使用 Linux 足够久,那么...
    99+
    2023-06-16
  • java中怎么用@RequiredArgsConstructor代替@Autowired
    这篇文章主要为大家展示了“java中怎么用@RequiredArgsConstructor代替@Autowired”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“java中怎么用@Required...
    99+
    2023-06-27
  • python3怎么利用pathlib替代os.path
    这篇文章主要介绍“python3怎么利用pathlib替代os.path”,在日常操作中,相信很多人在python3怎么利用pathlib替代os.path问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”pyt...
    99+
    2023-06-30
  • 怎么使用systemd timer替代cronjob
    今天小编给大家分享一下怎么使用systemd timer替代cronjob的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一...
    99+
    2023-07-05
  • PL/SQL在MySQL中的替代方案
    PL/SQL是Oracle数据库中常用的过程化编程语言,用于编写存储过程、触发器、函数等数据库对象。在MySQL数据库中,并没有直接的PL/SQL支持,但可以通过使用存储过程和触发器来...
    99+
    2024-03-15
    sql 数据库 oracle sql语句
  • win7怎么用键盘代替鼠标
    今天小编给大家分享一下win7怎么用键盘代替鼠标的相关知识点,内容详细,逻辑清晰,相信大部分人都还太了解这方面的知识,所以分享这篇文章给大家参考一下,希望大家阅读完这篇文章后有所收获,下面我们一起来了解一下吧。win7用键盘代替鼠标方法首先...
    99+
    2023-07-01
  • Elasticsearch的DSL操作命令怎么使用
    Elasticsearch的DSL操作命令可以通过发送HTTP请求来执行。以下是一些常见的DSL操作命令及其使用方法: 查询所有...
    99+
    2023-10-21
    DSL
  • sql中怎么用replace替换多个字符
    在SQL中可以使用REPLACE()函数来替换一个字符或多个字符,语法如下: REPLACE(string, old_substri...
    99+
    2024-04-09
    sql
  • sql中怎么使用replace替换多个字符
    在SQL中,可以使用`REPLACE`函数来替换多个字符。语法如下:```sqlREPLACE(string, old_su...
    99+
    2023-09-29
    SQL
  • SQL中有什么写法可以替代Like语句
    本篇内容介绍了“SQL中有什么写法可以替代Like语句”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成! ...
    99+
    2024-04-02
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作