Elasticsearch提供了一个非常全面和强大的REST API,可以使用它与集群进行交互。咱们来玩一下。
环境搭建
搭建好es
和kinaba
,可以访问通过ip:9200
查看es
的信息,ip:5601
打开kinaba
。
_cat命令
健康检查_cat/health
curl localhost:9200/_cat/health?v
集群的健康状态有绿色(green
)、黄色(yellow
),红色(red
)三种:
- 绿色:一切正常(集群功能全部可用)。
- 黄色:所有数据都可用,但某些副本尚未分配(集群完全正常工作)。
- 红色:由于某些原因,某些数据不可用(集群只有部分功能正常工作)。
节点列表_cat/nodes
curl localhost:9200/_cat/nodes?v
可以看到一个名为442a880a3b43
的节点,它是当前集群中唯一的节点。
集群的索引列表_cat/nodes
- health:索引的状态;
- index:索引名;
- docs.count:文档的个数;
其他命令
-
_cat/master
#查看master节点信息 -
_cat/segments
#查看各index的segment详细信息,包括segment名, 所属shard, 内存(磁盘)占用大小, 是否刷盘 -
_cat/count
#查看当前集群的doc数量
mapping映射
实操es数据之前,来了解一下es中的mapping映射
ES
中映射可以分为动态映射和静态映射
动态映射:在关系数据库中,需要事先创建数据库,然后在该数据库下创建数据表,并创建表字段、类型、长度、主键等,最后才能基于表插入数据。而Elasticsearch
中不需要定义Mapping
映射(即关系型数据库的表、字段等),在文档写入Elasticsearch
时,会根据文档字段自动识别类型,这种机制称之为动态映射。
如
#创建文档(ES根据数据类型, 会自动创建映射)
PUT /es_db/_doc/1
{
"name": "Jack",
"sex": 1,
"age": 25,
"book": "java入门至精通",
"address": "广州小蛮腰"
}
#获取文档映射
GET /es_db/_mapping
静态映射:静态映射是在Elasticsearch
中也可以事先定义好映射,包含文档的各字段类型、分词器等,这种方式称之为静态映射。
# 设置文档映射
PUT /es_db
{
"mappings":{
"properties":{
"name":{"type":"keyword","index":true,"store":true},
"sex":{"type":"integer","index":true,"store":true},
"age":{"type":"integer","index":true,"store":true},
"book":{"type":"text","index":true,"store":true},
"address":{"type":"text","index":true,"store":true}
}
}
}
# 根据静态映射创建文档
PUT /es_db/_doc/1
{
"name": "Jack",
"sex": 1,
"age": 25,
"book": "elasticSearch入门至精通",
"address": "广州车陂"
}
数据类型
字符串string
:,string类型包含text
和 keyword
。text
:该类型被用来索引长文本
,在创建索引前会将这些文本进行分词,转化为词的组合,建立索引;允许es
来检索这些词,text
类型不能用来排序和聚合。keyword
:该类型不能分词,可以被用来检索过滤、排序和聚合
,keyword
类型不可用text
进行分词模糊检索。
数值型:long、integer、short、byte、double、float
。
日期型:date
。
布尔型:boolean
。
ES数据操作
ES
是面向文档的,这意味着它可以存储整个对象或文档。然而它不仅仅是存储,还会索引(index)
每个文档的内容使之可以被搜索。在ES
中,你可以对文档(而非成行成列的数据)进行索引、搜索、排序、过滤。ES
使用JSON
作为文档序列化格式。
以kinaba的为例,索引的CURD
索引
- 创建索引
- 格式: PUT /索引名称
- 举例: PUT /es_db
#创建索引时可以设置分片数和副本数
PUT /es_db
{
"settings" : {
"number_of_shards" : 3,
"number_of_replicas" : 2
}
}
- 查询索引所有文档
- 格式: GET /索引名称
- 举例: GET /es_db
- 删除索引
- 格式: DELETE /索引名称
- 举例: DELETE /es_db
添加文档
- 格式: PUT/POST /索引名称/[_doc/_created]/id
_Create -如果ID已经存在,会失败
注意:POST
和PUT
都能起到创建/更新
的作用,PUT
需要确定id才能进行更新/创建,而POST
如果不写id
就由ES
生成一个唯一id进行创建新文档,如果填了id那就针对这个id的文档进行创建/更新。
修改文档
- 格式: ** [PUT | POST] /索引名称/_doc/id**
如果文档存在,现有文档会被删除,新的文档会被索引。
使用_update部分更新,格式: POST /索引名称/_update/id
并发场景下修改文档
_seq_no
和_primary_term
是对_version
的优化,7.X版本的ES默认使用这种方式控制版本,所以当在高并发环境下使用乐观锁机制修改文档时,要带上当前文档的_seq_no
和_primary_term
进行更新:
删除文档
- 格式: DELETE /索引名称/类型/id
- 举例: DELETE /es_db/_doc/1
查询某条文档
- 格式: GET /索引名称/类型/id
- 举例: GET /es_db/_doc/2
query string seach
- 查询当前类型中的所有文档
_search
, 类型可以写可不写。
- 格式: GET /索引名称/[/类型]/_search
- 举例: GET /es_db/_doc/_search 相当于
sql
:SQL: select * from _doc
- 条件查询:使用“q”指定查询字符串, 如要查询等于22岁的
_search?q=age:22
- 格式: GET /索引名称/[/类型]/_search?q=条件属性:值
- GET es_db/_doc/_search?q=age:22
- 范围查询 :如要查询age在25至26岁之间的
- 格式: GET /索引名称[/类型]/_search?q=条件属性[值1 TO 值2] 注意: TO 必须为大写
- 举例: GET /es_db/_doc/_search?q=age[22 TO 26]
- 分页查询
from=*&size=*
- 格式: GET /索引名称[/类型]/_search?q=条件属性[值1 TO 值2]&from=0&size=1
- 举例: GET /es_db/_doc/_search?q=age[25 TO 26]&from=0&size=1相当于
select * from _doc where age between 22 and 26 limit 0, 1
- 对查询结果排序
sort=字段:desc/asc
- 格式: GET /索引名称[/类型]/_search?sort=条件属性:desc/asc
- GET /es_db/_search?sort=age:desc
- 对查询结果只输出某些字段
_source=字段,字段
- 格式: GET /索引名称[/类型]/_search?_source=属性
- GET /es_db/_search?_source=name&sort=age:desc
注意:查询的时候带上类型会出现Deprecation: [types removal] Specifying types in search requests is deprecated.
,es7
版本将type
弃用导致的。
query DSL
DSL由叶子查询子句和复合查询子句两种子句组成。
无查询条件:无查询条件是查询所有,默认是查询所有的,或者使用match_all
表示所有。
GET /es_db/_search
{
"query": {
"match_all": {}
}
}
模糊查询
主要是针对文本类型的字段,文本类型的字段会对内容进行分词,对查询时,也会对搜索条件进行分词,然后通过倒排索引查找到匹配的数据。
-
match
: 通过match关键词模糊匹配条件内容 -
prefix
: 前缀匹配 -
regexp
: 通过正则表达式来匹配数据
#模糊查询 单字段
GET /es_db/_search
{
"query": {
"match": {
"address": "中国"
}
},
"from":0,
"size": 1,
"_source":["name","age","sex"]
}
#查询address和name中包含caicai 多字段
GET /es_db/_search
{
"query":{
"multi_match":{
"query":"caicai",
"fields":["address","name"]
}
}
}
# 前缀类型
GET /es_db/_search
{
"query": {
"prefix": {
"name": {
"value": "a"
}
}
}
}
# match_phrase 完全匹配
GET /es_db/_search
{
"query":{
"match_phrase":{
"address":"中国人"
}
}
}
精确匹配
-
term
: 单个条件相等 -
terms
: 单个字段属于某个值数组内的值 -
range
: 字段属于某个范围内的值 -
exists
: 某个字段的值是否存在 -
ids
: 通过ID批量查询
match
和term
的区别
-
match
:模糊匹配,需要指定字段名,但是输入会进行分词,比如"hello world"
会进行拆分为hello
和world
,然后匹配,如果字段中包含hello
或者world
,或者都包含的结果都会被查询出来,也就是说match
是一个部分匹配的模糊查询。查询条件相对来说比较宽松。 -
term
: 这种查询和match
在有些时候是等价的,比如我们查询单个的词hello
,那么会和match
查询结果一样,但是如果查询"hello world"
,结果就相差很大,因为这个输入不会进行分词,就是说查询的时候,是查询字段分词结果中是否有"hello world"
的字样,而不是查询字段中包含"hello world"
的字样。当保存
数据"hello world"
时,elasticsearch
会对字段内容进行分词,"hello world"
会被分成hello
和world
,不存在"hello world"
,因此这里的查询结果会为空。这也是term查询和match的区别。
## 根据sex精准匹配
GET /es_db/_search
{
"query": {
"term": {
"sex": {
"value": "0"
}
}
}
}
# 年龄10-25之间
GET /es_db/_search
{
"query": {
"range": {
"age": {
"gte": 10,
"lte": 25
}
}
}
}
# 年龄name叫caicai或者admin的人
GET /es_db/_search
{
"query": {
"terms": {
"name": [
"caicai",
"admin"
]
}
}
}
组合条件查询
组合条件查询是将叶子条件查询语句进行组合而形成的一个完整的查询条件。
- bool : 各条件之间有and,or或not的关系。
-
must
: 各个条件都必须满足,即各条件是and
的关系。
-
should
: 各个条件有一个满足即可,即各条件是or
的关系。 -
must_not
: 不满足所有条件,即各条件是not
的。
关系 -
filter
: 不计算相关度评分,它不计算_score
。即相关度评分,效率更高。
constant_score : 不计算相关度评分must/filter/shoud/must_not
等的子条件是通过term/terms/range/ids/exists/match
等叶子条件为参数的。
# 查询地址包含中国,而且年龄在10-25之间的人
GET /es_db/_search
{
"query": {
"bool": {
"must": [
{
"match": {
"address": "中国"
}
},
{
"range": {
"age": {
"gte": 10,
"lte": 25
}
}
}
]
}
}
}
#查询address有中国的
GET /es_db/_search
{
"query":{
"match":{
"address":"中"
}
}
}
##高亮
GET /es_db/_search
{
"query":{
"bool":{
"filter":{
"match":{
"address":"中"
}
}
}
},
"highlight": {
"fields": {"address": {}}
}
}
查询DSL(query DSL
)和过滤DSL(filter DSL
)
- query DSL:在查询上下文中,查询会回答这个问题——“这个文档匹不匹配这个查询,它的相关度高么?”如何验证匹配很好理解,如何计算相关度呢?
ES
中索引的数据都会存储一个_score
分值,分值越高就代表越匹配。另外关于某个搜索的分值计算还是很复杂的,因此也需要一定的时间。 - filter DSL:在过滤器上下文中,查询会回答这个问题——“这个文档匹不匹配?”答案很简单,是或者不是。它不会去计算任何分值,也不会关心返回的排序问题,因此效率会高一点。过滤上下文 是在使用
filter
参数时候的执行环境,比如在bool查询中使用must_not
或者filter
另外,经常使用过滤器,ES
会自动的缓存过滤器的内容,这对于查询来说,会提高很多性能。
es搜索相关参数
-
took
:耗费的时间,单位毫秒。 - ·time_out·:是否超时。
-
_shards
:几个分片。 -
hit.total
:查询结果的数量。 -
hit.max_score
:匹配的分数。 -
hit.hits
:具体结果
ElasticSearch文档批量操作
- 批量操作可以减少网络连接所产生的开销,提升性能;
- 支持在一次API调用中,对不同的索引进行操作;
- 可以在URI中指定Index,也可以在请求的Payload中进行;
- 操作中单条操作失败,并不会影响其他操作;
- 返回结果包括了每一条操作执行的结果;
批量写入
批量对文档进行写操作是通过_bulk
的API来实现的
- 请求方式:
POST
- 请求地址:
_bulk
- 请求参数:通过_bulk操作文档,一般至少有两行参数(或偶数行参数)
- 第一行参数为**指定操作的类型(
create,index,delete和update
)**及操作的对象(index,type和id) - 第二行参数才是操作的数据
批量创建文档create
POST _bulk
{"create":{"_index":"article", "_type":"_doc", "_id":3}}
{"id":3,"title":"java","content":"java java","tags":["java", "面向对象"],"create_time":1554015482530}
{"create":{"_index":"article", "_type":"_doc", "_id":4}}
{"id":4,"title":"golang","content":"golang","tags":["golang", "面向对象"],"create_time":1554015482530}
普通创建或全量替换index
POST _bulk
{“index”:{“_index”:“article”, “_type”:“_doc”, “_id”:3}}
{“id”:3,“title”:“golang”,“content”:“golang”,“tags”:[“golang”, “面向对象”],“create_time”:1554015482530}
{“index”:{“_index”:“article”, “_type”:“_doc”, “_id”:4}}
{“id”:4,“title”:“java”,“content”:“java”,“tags”:[“java”, “面向对象”],“create_time”:1554015482530}
批量删除delete
POST _bulk
{"delete":{"_index":"article", "_type":"_doc", "_id":3}}
{"delete":{"_index":"article", "_type":"_doc", "_id":4}}
批量修改update
POST _bulk
{"update":{"_index":"article", "_type":"_doc", "_id":3}}
{"doc":{"title":"ES大法必修内功"}}
{"update":{"_index":"article", "_type":"_doc", "_id":4}}
{"doc":{"create_time":1554018421008}}
批量读取es
的批量查询可以使用mget
和msearch
两种。其中mget
是需要我们知道它的id,可以指定不同的index
,也可以指定返回值source
。msearch
可以通过字段查询来进行一个批量的查找。
最后
实操笔记
#创建索引
PUT /es_db
# 删除索引
DELETE /es_db
# 查询索引
GET /es_db
PUT /es_db/_doc/1
{
"name":"张三",
"sex":1,
"age":25,
"address":"中国",
"remark":"java developer"
}
#创建文档,ES生成id
POST /es_db/_doc
{
"name": "张三",
"sex": 1,
"age": 25,
"address": "广州天河公园",
"remark": "java developer"
}
# 修改id为1的数据
PUT /es_db/_doc/1
{
"name":"caicai",
"sex":1,
"age":25,
"address":"中国",
"remark":"golang"
}
# 部分更新:在原有文档上更新
# Update -文档必须已经存在,更新只会对相应字段做增量修改
POST /es_db/_update/1
{
"doc": {
"age": 28
}
}
#查询文档
GET /es_db/_doc/1
## 并发修改
POST /es_db/_doc/2?if_seq_no=21&if_primary_term=6
{
"name": "李四xxx"
}
# 删除文档
DELETE /es_db/_doc/1
PUT /es_db/_doc/2
{
"name":"李四",
"sex":1,
"age":28,
"address":"中国香港",
"remark":"java assistant"
}
PUT /es_db/_doc/3
{
"name":"rod",
"sex":0,
"age":26,
"address":"北京",
"remark":"php developer"
}
PUT /es_db/5
{
"name":"admin",
"sex":0,
"age":22,
"address":"长沙",
"remark":"pythonassistant"
}
# 查询文档
GET /es_db/_doc/2
GET es_db/_doc/_search?q=age:22
# 同上 GET es_db/_search?q=age:22
GET /es_db/_doc/_search?q=age[22 TO 26]
# 同上 GET /es_db/_search?q=age[22 TO 26]
GET /es_db/_doc/_search?q=age[22 TO 26]&from=0&size=1
# 同上 GET /es_db/_search?q=age[22 TO 26]&from=0&size=1
GET /es_db/_search?sort=age:desc
GET /es_db/_search?_source=name&sort=age:desc
###########################################
# 索引的信息 包含字段的类型等
GET /es_db/
# 全部
GET /es_db/_search
{
"query": {
"match_all": {}
}
}
#模糊查询 单字段
GET /es_db/_search
{
"query": {
"match": {
"address": "中国"
}
},
"from":0,
"size": 1,
"_source":["name","age","sex"]
}
#查询address和name中包含caicai
GET /es_db/_search
{
"query":{
"multi_match":{
"query":"caicai",
"fields":["address","name"]
}
}
}
# 前缀类型
GET /es_db/_search
{
"query": {
"prefix": {
"name": {
"value": "a"
}
}
}
}
GET /es_db/_search
{
"query":{
"match_phrase":{
"address":"中国人"
}
}
}
## 根据sex精准匹配
GET /es_db/_search
{
"query": {
"term": {
"sex": {
"value": "0"
}
}
}
}
# 年龄10-25之间
GET /es_db/_search
{
"query": {
"range": {
"age": {
"gte": 10,
"lte": 25
}
}
}
}
# 年龄name叫caicai或者admin的人
GET /es_db/_search
{
"query": {
"terms": {
"name": [
"caicai",
"admin"
]
}
}
}
# 查询地址包含中国,而且年龄在10-25之间的人
GET /es_db/_search
{
"query": {
"bool": {
"must": [
{
"match": {
"address": "中国"
}
},
{
"range": {
"age": {
"gte": 10,
"lte": 25
}
}
}
]
}
}
}
GET /es_db/_search
{
"query":{
"match":{
"address":"中"
}
}
}
## 高亮
GET /es_db/_search
{
"query":{
"bool":{
"filter":{
"match":{
"address":"中"
}
}
}
},
"highlight": {
"fields": {"address": {}}
}
}
######################################################
# 批量
## 批量创建
POST _bulk
{"create":{"_index":"article", "_type":"_doc", "_id":3}}
{"id":3,"title":"java","content":"java java","tags":["java", "面向对象"],"create_time":1554015482530}
{"create":{"_index":"article", "_type":"_doc", "_id":4}}
{"id":4,"title":"golang","content":"golang","tags":["golang", "面向对象"],"create_time":1554015482530}
POST _bulk
{"index":{"_index":"article", "_type":"_doc", "_id":3}}
{"id":3,"title":"golang","content":"golang","tags":["golang", "面向对象"],"create_time":1554015482530}
{"index":{"_index":"article", "_type":"_doc", "_id":4}}
{"id":4,"title":"java","content":"java","tags":["java", "面向对象"],"create_time":1554015482530}
#批量删除delete
POST _bulk
{"delete":{"_index":"article", "_type":"_doc", "_id":3}}
{"delete":{"_index":"article", "_type":"_doc", "_id":4}}
##批量修改
POST _bulk
{"update":{"_index":"article", "_type":"_doc", "_id":3}}
{"doc":{"title":"ES大法必修内功"}}
{"update":{"_index":"article", "_type":"_doc", "_id":4}}
{"doc":{"create_time":1554018421008}}
#可以通过ID批量获取不同index和type的数据
GET _mget
{
"docs": [
{
"_index": "es_db",
"_id": 1
},
{
"_index": "article",
"_id": 4
}
]
}
#可以通过ID批量获取es_db的数据
GET /es_db/_mget
{
"docs": [
{
"_id": 1
},
{
"_id": 4
}
]
}
#简化后
GET /es_db/_mget
{
"ids":["1","2"]
}