ElasticSearch(五):Mapping和常见字段类型

学习课程链接《Elasticsearch核心技术与实战》

## 什么是Mapping
* Mapping类似数据库中的schema的定义,作用如下:
- 定义索引中的字段的名称;
- 定义字段的数据类型,例如字符串、数字、日期、布尔等;
- 对每个字段进行倒排索引的相关配置(Analyzed or Not Analyzed,Analyzer);
* Mapping 会把JSON文旦映射成Lucene所需要的扁平格式。
* 一个Mapping属于一个索引的Type:
- 每个文档都属于一个Type;
- 一个Tpye有一个Mapping定义;
- 7.0开始,不需要再Mapping定义中指定type信息;

## 字段的数据类型
* 简单类型
- Text
- Date
- Integer/Long/Floating
- Boolean
- IP4&IP6
- Keyword
* 复杂类型
- 对象类型
- 嵌套类型
* 特殊类型(地理信息)
- geo_point&geo_shape、percolator

## 什么是Dynamic Mapping
* 在写入文档的时候,如果索引不存在,则会自动创建索引;
* Dynamic Mapping机制,可以无需手动定义Mapping,ElasticSearch会自动根据文档信息,推算出字段的类型;
* 但是有时候推算的可能不对,例如地理位置信息;
* 当类型设置的不对时,会导致一些功能无法正常运行,比如范围内的Range查询;

## 类型的自动识别
JSON类型|Elasticsearch类型
---|---
字符串|匹配日期格式,设置成Date;匹配数字设置成Float或者Long,该选项默认关闭;设置为Text,并且增加keyword子字段
布尔值|Boolean
浮点数|Float
整数|Long
对象|Object
数组|由第一个非空数的类型所决定
空值|忽略

```
#写入文档,查看 Mapping
PUT mapping_test/_doc/1
{
"firstName":"Chan",
"loginDate":"2018-07-24T10:29:48.103Z",
"uid" : "123",
"isVip" : false,
"isAdmin": "true",
"age":19,
"heigh":180
}

Delete index

DELETE mapping_test

查看 Dynamic Mapping文件

GET mapping_test/_mapping


查看 Dynamic Mapping返回结果

{

"mapping_test" : {

"mappings" : {

"properties" : {

"age" : {

"type" : "long" # "age":19,设置为long

},

"firstName" : {

"type" : "text", # "firstName":"Chan",设置为Text,并且增加keyword子字段

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

},

"heigh" : {

"type" : "long" #"heigh":180设置为long

},

"isAdmin" : {

"type" : "text", #"isAdmin": "true",设置为Text,并且增加keyword子字段

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

},

"isVip" : {

"type" : "boolean" #"isVip" : false,设置为boolean

},

"loginDate" : {

"type" : "date" #"loginDate":"2018-07-24T10:29:48.103Z",设置为Date

},

"uid" : {

"type" : "text", # "uid" : "123",设置为Text,并且增加keyword子字段,匹配数字设置成Float或者Long,该选项默认关闭;

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

}

}

}

}

}



<br/>
## 能否更改 Mapping 的字段类型
分两种情况:
* 新增加字段
- Dynamic设置为true时,一旦有新增字段的文档写入,Mapping也同时被更新;
- Dynamic设置为false时,Mapping不会被更新,新增字段的数据无法被索引,但是信息会出现在_source中;
- Dynamic设置为strict时,文档写入失败;
* 对已有字段,一旦已有数据写入,就不在支持修改字段定义
- Lucene实现的倒排索引,一旦生成后,就不允许修改
- 如果希望修改字段类型,必须Reindex API,重建索引
- 如果修改了字段的数据类型,会导致已被索引的数据无法被搜索 <br/>
## 控制Dynamic Mappings
dynamic|true|false|strict
---|---|---|---
文档可索引|YES|YES|NO
字段可索引|YES|NO|NO
Mapping被更新|YES|NO|NO
<br/>
* 当dynamic被设置成false时,存在新增字段数据写入,该数据可以被索引,但新增字段被丢弃
* 当dynamic被设置成strict时,数据写入直接出错

1.默认Mapping支持dynamic,写入的文档中加入新的字段

PUT dynamic_mapping_test/_doc/1

{

"newField":"someValue"

}


2.该字段可以被搜索,数据也在_source中出现

POST dynamic_mapping_test/_search

{

"query":{

"match":{

"newField":"someValue"

}

}

}

返回结果:

{

"took" : 5,

"timed_out" : false,

"_shards" : {

"total" : 1,

"successful" : 1,

"skipped" : 0,

"failed" : 0

},

"hits" : {

"total" : {

"value" : 1,

"relation" : "eq"

},

"max_score" : 0.2876821,

"hits" : [

{

"_index" : "dynamic_mapping_test",

"_type" : "_doc",

"_id" : "1",

"_score" : 0.2876821,

"_source" : {

"newField" : "someValue"

}

}

]

}

}


3.修改为dynamic false

PUT dynamic_mapping_test/_mapping

{

"dynamic": false

}


4.新增 anotherField

PUT dynamic_mapping_test/_doc/10

{

"anotherField":"someValue"

}


5.该字段不可以被搜索,因为dynamic已经被设置为false

POST dynamic_mapping_test/_search

{

"query":{

"match":{

"anotherField":"someValue"

}

}

}

返回结果:

{

"took" : 657,

"timed_out" : false,

"_shards" : {

"total" : 1,

"successful" : 1,

"skipped" : 0,

"failed" : 0

},

"hits" : {

"total" : {

"value" : 0,

"relation" : "eq"

},

"max_score" : null,

"hits" : [ ]

}

}


6.修改为strict

PUT dynamic_mapping_test/_mapping

{

"dynamic": "strict"

}


7.写入数据出错,HTTP Code 400

PUT dynamic_mapping_test/_doc/12

{

"lastField":"value"

}

返回结果:

{

"error": {

"root_cause": [

{

"type": "strict_dynamic_mapping_exception",

"reason": "mapping set to strict, dynamic introduction of [lastField] within [_doc] is not allowed"

}

],

"type": "strict_dynamic_mapping_exception",

"reason": "mapping set to strict, dynamic introduction of [lastField] within [_doc] is not allowed"

},

"status": 400

}



<br/>
## 如何定义一个 Mapping

PUT index_name

{

"mappings":{

"properties":{

//define your mappings here

}

}

}


* 可以参考API手册,纯手写;
* 为了减少输入的工作量,减少出错概率,可以依照以下步骤:
- 创建一个临时的index,写入一些样本数据;
- 通过访问Mapping API获取该临时文件的动态Mapping定义;
- 修改后,使用该配置创建你的索引
- 删除临时索引 <br/>
## Mapping的一些配置
* ` index`控制当前字段是否被索引,默认为`true`。如果设置成`false`,该字段不可被搜索。
* `index_options`可以控制倒排索引记录的内容,有四种不同级别的配置:
- `docs`记录 doc id
- `freqs`记录 doc id / term frequencies
- `positions`记录 doc id / term frequencies / term position
- `offects`记录 doc id / term frequencies / term position / character offects
* Text类型默认记录`positions`,其他默认为 `docs`。记录的类容越多,占用存储空间越大。
* ` null_value`控制需要对Null值实现搜索;只有Keyword类型支持设定null_value。
* ` copy_to`满足一些特定的搜索需求,` copy_to`将字段的数值拷贝到目标字段,实现类似`_all`的作用,`_all`在ES7中被` copy_to`所替代,` copy_to`的目标字段不出现在_source中。
* Elasticsearch中不提供专门的数组类型。但是任何字段,都可以包含多个相同类类型的数值。

1.设置 index 为 false

DELETE users

PUT users

{

"mappings" : {

"properties" : {

"firstName" : {

"type" : "text"

},

"lastName" : {

"type" : "text"

},

"mobile" : {

"type" : "text",

"index": false

}

}

}

}

插入数据

PUT users/_doc/1

{

"firstName":"Ruan",

"lastName": "Yiming",

"mobile": "12345678"

}

查询

POST /users/_search

{

"query": {

"match": {

"mobile":"12345678" #该字段不可被搜索

}

}

}

查询返回结果:

{

"error": {

"root_cause": [

{

"type": "query_shard_exception",

"reason": "failed to create query: {\n "match" : {\n "mobile" : {\n "query" : "12345678",\n "operator" : "OR",\n "prefix_length" : 0,\n "max_expansions" : 50,\n "fuzzy_transpositions" : true,\n "lenient" : false,\n "zero_terms_query" : "NONE",\n "auto_generate_synonyms_phrase_query" : true,\n "boost" : 1.0\n }\n }\n}",

"index_uuid": "1oB9dwY2TPq-9QjiaMaU7g",

"index": "users"

}

],

"type": "search_phase_execution_exception",

"reason": "all shards failed",

"phase": "query",

"grouped": true,

"failed_shards": [

{

"shard": 0,

"index": "users",

"node": "u-4S1mfbQiuA1Bqe-wfPJQ",

"reason": {

"type": "query_shard_exception",

"reason": "failed to create query: {\n "match" : {\n "mobile" : {\n "query" : "12345678",\n "operator" : "OR",\n "prefix_length" : 0,\n "max_expansions" : 50,\n "fuzzy_transpositions" : true,\n "lenient" : false,\n "zero_terms_query" : "NONE",\n "auto_generate_synonyms_phrase_query" : true,\n "boost" : 1.0\n }\n }\n}",

"index_uuid": "1oB9dwY2TPq-9QjiaMaU7g",

"index": "users",

"caused_by": {

"type": "illegal_argument_exception",

"reason": "Cannot search on field [mobile] since it is not indexed." #错误原因

}

}

}

]

},

"status": 400

}


设定Null_value

DELETE users

PUT users

{

"mappings" : {

"properties" : {

"firstName" : {

"type" : "text"

},

"lastName" : {

"type" : "text"

},

"mobile" : {

"type" : "keyword",

"null_value": "NULL"

}

  }
}

}

插入数据

PUT users/_doc/1

{

"firstName":"Ruan",

"lastName": "Yiming",

"mobile": null

}

插入数据

PUT users/_doc/2

{

"firstName":"Ruan2",

"lastName": "Yiming2"

}

查询

GET users/_search

{

"query": {

"match": {

"mobile":"NULL"

}

}

}

查询返回结果:

{

"took" : 1,

"timed_out" : false,

"_shards" : {

"total" : 1,

"successful" : 1,

"skipped" : 0,

"failed" : 0

},

"hits" : {

"total" : {

"value" : 1,

"relation" : "eq"

},

"max_score" : 0.2876821,

"hits" : [

{

"_index" : "users",

"_type" : "_doc",

"_id" : "1",

"_score" : 0.2876821,

"_source" : {

"firstName" : "Ruan",

"lastName" : "Yiming",

"mobile" : null

}

}

]

}

}

设置 Copy to

DELETE users

PUT users

{

"mappings": {

"properties": {

"firstName":{

"type": "text",

"copy_to": "fullName"

},

"lastName":{

"type": "text",

"copy_to": "fullName"

}

}

}

}

插入数据

PUT users/_doc/1

{

"firstName":"Ruan",

"lastName": "Yiming"

}

查询方法1

GET users/_search?q=fullName:(Ruan Yiming)

查询方法2

POST users/_search

{

"query": {

"match": {

"fullName":{

"query": "Ruan Yiming",

"operator": "and"

}

}

}

}

查询返回结果:

{

"took" : 1,

"timed_out" : false,

"_shards" : {

"total" : 1,

"successful" : 1,

"skipped" : 0,

"failed" : 0

},

"hits" : {

"total" : {

"value" : 1,

"relation" : "eq"

},

"max_score" : 0.5753642,

"hits" : [

{

"_index" : "users",

"_type" : "_doc",

"_id" : "1",

"_score" : 0.5753642,

"_source" : {

"firstName" : "Ruan",

"lastName" : "Yiming"

}

}

]

}

}


数组类型

PUT users/_doc/1

{

"name":"twobirds",

"interests":["reading","music"]

}

GET users/_mapping

返回Mapping结果:

{

"users" : {

"mappings" : {

"properties" : {

"firstName" : {

"type" : "text",

"copy_to" : [

"fullName"

]

},

"fullName" : {

"type" : "text",

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

},

"interests" : {

"type" : "text", #数组类型,根据数组里数据类型配置

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

},

"lastName" : {

"type" : "text",

"copy_to" : [

"fullName"

]

},

"name" : {

"type" : "text",

"fields" : {

"keyword" : {

"type" : "keyword",

"ignore_above" : 256

}

}

}

}

}

}

}



<br/>

ElasticSearch(五):Mapping和常见字段类型的更多相关文章

  1. Dynamic Mapping和常见字段类型

    原文:Dynamic Mapping和常见字段类型 版权声明:本文为博主原创文章,遵循CC 4.0 BY-SA版权协议,转载请附上原文出处链接和本声明. 本文链接:https://blog.csdn. ...

  2. elasticsearch入门使用(二) Mapping + field type字段类型

    Elasticsearch Reference [6.2] » Mapping 参考官方英文文档 https://www.elastic.co/guide/en/elasticsearch/refer ...

  3. ElasticSearch Mapping中的字段类型

    1)string: 默认会被分词 2)数字类型主要如下几种: long:64位存储  integer:32位存储  short:16位存储  byte:8位存储  double:64位双精度存储  f ...

  4. 转载:Oracle常见字段类型

    转载节选自:https://bbs.csdn.net/topics/220059184 数据类型 参数 描述 char(n) n=1 to 2000字节 定长字符串,n字节长,如果不指定长度,缺省为1 ...

  5. ElasticSearch(5)-Mapping

    一.Mapping概述 映射 为了能够把日期字段处理成日期,把数字字段处理成数字,把字符串字段处理成全文本(Full-text)或精确的字符串值,Elasticsearch需要知道每个字段里面都包含了 ...

  6. 【转】elasticsearch中字段类型默认显示{ "foo": { "type": "text", "fields": { "keyword": {"type": "keyword", "ignore_above": 256} }

    官方原文链接:https://www.elastic.co/cn/blog/strings-are-dead-long-live-strings 转载原文连接:https://segmentfault ...

  7. Elasticsearch支持的字段类型

    es支持下列简单的字段类型: String: string Whole number: byte, short, integer, long Floating point: float, double ...

  8. ES系列六、ES字段类型及ES内置analyzer分析

    一.背景知识 在Es中,字段的类型很关键: 在索引的时候,如果字段第一次出现,会自动识别某个类型,这种规则之前已经讲过了. 那么如果一个字段已经存在了,并且设置为某个类型.再来一条数据,字段的数据不与 ...

  9. ElasticStack学习(七):ElasticSearch之Mapping初探

    一.Mapping的概念 1.Mapping类似于数据库中的Schema的定义,作用如下: 1)定义索引中的字段的名称: 2)定义字段的数据类型,例如字符串.数字.日期.布尔等: 3)对每个字段进行倒 ...

随机推荐

  1. 设计模式----创建型模式之工厂模式(FactoryPattern)

    工厂模式主要分为三种简单工厂模式.工厂方法模式.抽象工厂模式三种.顾名思义,工厂主要是生产产品,作为顾客或者商家,我们不考虑工厂内部是怎么一个流程,我们要的是最终产品.将该种思路放在我们面向对象开发实 ...

  2. 最简单的JS实现json转csv

    工作久了,总会遇到各种各样的数据处理工作,比如同步数据,初始化一些数据,目前比较流行的交互数据格式就是JSON,可是服务器中得到的JSON数据如果提供给业务人员看的话可能会非常不方便,这时候,转成CS ...

  3. 04-09 XgBoost算法

    目录 XgBoost算法 一.XgBoost算法学习目标 二.XgBoost算法详解 2.1 XgBoost算法参数 2.2 XgBoost算法目标函数 2.3 XgBoost算法正则化项 2.4 X ...

  4. 重磅发布 | 全球首个云原生应用标准定义与架构模型 OAM 正式开源

    作者: OAM 项目负责人 导读:2019 年 10 月 17 日,阿里巴巴合伙人.阿里云智能基础产品事业部总经理蒋江伟(花名:小邪)在 Qcon 上海重磅宣布,阿里云与微软联合推出开放应用模型 Op ...

  5. Elasticsearch Java API 很全的整理

    Elasticsearch 的API 分为 REST Client API(http请求形式)以及 transportClient API两种.相比来说transportClient API效率更高, ...

  6. springboot 自定义LocaleResolver切换语言

    springboot 自定义LocaleResolver切换语言 我们在做项目的时候,往往有很多项目需要根据用户的需要来切换不同的语言,使用国际化就可以轻松解决. 我们可以自定义springboor中 ...

  7. tcpdump使用详情及案例

    转载http://starsliao.blog.163.com/blog/static/89048201062333032563/ TCPdump抓包命令tcpdump是一个用于截取网络分组,并输出分 ...

  8. 前端开发 JavaScript 干货知识点汇总

    很多初学的朋友经常问我,前端JavaScript都需要学习哪些东西呀?哪些是JavaScript的重点知识啊? 其实做前端开发工程师,所有的知识点都是我们学习必备的东西,只有扎实的技术基础才是高薪的关 ...

  9. Java学习笔记之基础语法(数据类型)

    8种基本数据类型    整型:   byte[1字节]          short[2字节]        int[4字节]         long[8字节]      1,四种整型之间的区别:申 ...

  10. 利用python模拟菜刀反弹shell绕过限制

    有的时候我们在获取到目标电脑时候如果对方电脑又python 编译环境时可以利用python 反弹shell 主要用到python os库和sokect库 这里的服务端在目标机上运行 from sock ...