1200字范文,内容丰富有趣,写作的好帮手!
1200字范文 > MongoDB学习笔记之MongoDB数据模型

MongoDB学习笔记之MongoDB数据模型

时间:2019-04-26 04:08:03

相关推荐

MongoDB学习笔记之MongoDB数据模型

一、BSON协议与数据类型

JSON介绍

JSON是当今非常通用的一种跨语言Web数据交互格式,属于ECMAScript标准规范的一个子集。JSON(JavaScript Object Notation:JS对象简谱)即JavaScript对象表示法,它是JavaScript对象的一种文本表现形式。

作为一种轻量级的数据交换格式,JSON的可读性非常好,而且非常便于系统生成和解析,这些优势也让他逐渐取代了XML标准在WEB领域的地位,当今许多流行的Web应用开发框架,如SpringBoot都选择了JSON作为默认的数据编/解码格式。

JSON只定义了6种数据类型:

string: 字符串number : 数值object: JS的对象形式,用{key:value}表示,可嵌套array: 数组,JS的表示方式[value],可嵌套true/false: 布尔类型null: 空值

大多数情况下,使用JSON作为数据交互格式已经是理想的选择,但是JSON基于文本的解析效率并不是最好的,在某些场景下往往会考虑选择更合适的编/解码格式,一些做法如:

在微服务架构中,使用gRPC(基于Google的Protobuf)可以获得更好的网络利用率。分布式中间件、数据库,使用私有定制的TCP数据包格式来提供高性能、低延时的计算能力。

BSON介绍

BSON由10gen团队设计并开源,目前主要用于MongoDB数据库。BSON(Binary JSON)是二进制版本的JSON,其在性能有更优的表现。BSON在许多方面和JSON保持一致,同样也支持内嵌的文档对象和数组结构。二者最大的区别在于JSON是基于文本的,而BSON则是二进制(字节流)编/解码形式。在空间的使用上,BSON相比JSON并没有明显的优势。

MongoDB在文档存储、命令协议上都采用了BSON作为编/解码格式,主要具有如下优势:

类JSON的轻量级语义,支持简单清晰的嵌套、数组层次结构,可以实现模式灵活的文档结构。更高效的遍历,BSON在编码时会记录每个元素的长度,可以直接通过seek操作进行元素的内容读取,相对JSON解析来说,遍历速度更快。更丰富的数据类型,除了JSON的基本数据类型,BSON还提供了MongoDB所需的一些扩展类型,比如日期、二进制数据等,这更加方便数据的表示和操作。

BSON的数据类型

MongoDB中,一个BSON文档最大大小为16M,文档嵌套的级别不超过100

BSON Types — MongoDB Manual

$type操作符

$type操作符基于BSON类型来检索集合中匹配的数据类型,并返回结果

db.books.find({"title":{$type:2}})//或者db.books.finds({"title":{$type:"string"}})

二、日期类型

MongoDB的日期类型使用UTC(Coordinated Universql Time) 进行存储,也就是+0时区的时间。

db.dates.insert([{data1:Date()}.{data2:new Date()},{data3:ISODate()}])db.dates.find().pretty()

使用new Date与ISODate最终都会生成ISODate类型的字段(对应于UTC时间)

三、ObjectId生成器

MongoDB集合中所有的文档都有一个唯一的_id字段,作为集合的主键。在默认情况下,_id字段使用ObjectId类型,采用16进制编码格式,共12个字节。

为了避免文档的_id字段出现重复,ObjectId被定义为3个部分:

4字节表示Unix时间戳(秒)5字节表示随机数(机器号+进程号唯一)3字节表示计数器(初始化时随机)

大多数客户端驱动都会自行生成这个字段,比如MongoDB Java Driver会根据插入的文档是否包含_id字段来自动补充ObjectId对象。这样做不但提高了离散性,还可以降低MongoDB服务器端的计算压力。在ObjectId的组成中,5字节的随机数并没有明确定义,客户端可以采用机器号、进程号来实现:

生成一个新的ObjectId

x = ObjectId()

四、内嵌文档和数组

一个文档中可以包含作者的信息,包括作者名称、性别、家乡所在地,一个显著的优点是,当我们查询book文档的信息时,作者的信息也一并返回。

db.books.insert({title:"标题",author:{name:"yaya",gender:"男",hometown:"山东"}})

查询yaya的作品

db.books.find({"author.name":"yaya"})

修改yaya的家乡所在地

db.books.updateOne({"author.name":"yaya"},{$set:{"author.hometown":"中国"}}.)

数组

除了作者信息,文档中还包含了若干个标签,这些标签可以用来表示文档所包含的一些特征,如豆瓣读书中的标签(tag)

添加tags标签

db.books.updateOne({"author.name":"三毛"},{$set:{tags:["旅行","随笔","散文"]}})

查询数组元素

#会查询到所有的tagsdb.books.find({"author.name":"三毛"},{$push:{tags:"猎奇"}})

$push操作符可以配合其他操作符,一起实现不同的数组修改操作,比如和$each操作符配合可以用于添加多个元素

db.books.updateOne({"author.name":"三毛"},{$push:{tags:{"伤感","想象力"}}})

如果加上$slice操作符,那么只会保留经过切片后的元素

db.books.updateOne({"author.name":"三毛"},{$push:{tags:{$each:["伤感","想象力"],$slice:3}}}})

根据元素查询

#会查出所有包含伤感的文档db.books.finds({tags:"伤感"})#会查出所有同事包含“伤感”“想象力”的文档db.books.find({tags:{$all:["伤感","想象力"]}})

嵌套型的数组

数组元素可以是基本类型,也可以是内嵌的文档结构

{tags:[{tagKey:xxx,tagValue:xxxx},{tagKey:xxx,tagValue:xxxx}]}

这种结构非常灵活,一个很适合的场景就是商品的多属性表示

db.goods.insertMany([{name:"羽绒服",tags:[{tagkey:"size",tagValue:["M","L","XL","XXL","XXXL"]},{tagKey:"color",tagValue:["黑色","宝蓝"]},{tagKey:"style",tagValue:"韩风"}]},{name:"衬衣",tags:[{tagkey:"size",tagValue:["L","XL","XXL","XXXL"]},{tagKey:"color",tagValue:["黑色","宝蓝"]},{tagKey:"style",tagValue:"韩风"}]}])

以上的设计是一种常见的多值属性的做法,当我们需要根据属性进行检索时,需要用到$elementMatch操作符:

#筛选出color=黑色的商品信息db.goods.find({tags:{$elemMatch:{tagKey:"color",tagValue:"黑色"}}})

如果进行组合式的条件检索,则可以使用多个$elemMatch操作符:

#筛选出color=蓝色,并且size=XL的商品信息db.goods.find({tags:{$all:[{$elemMatch:{tagKey:"color",tagValue:"黑色"}},{$elemMatch:{tagKey:"size",tagValue:"XL"}}]}})

五、固定集合

固定集合(capped collection)是一种限定大小的集合,其中capped是覆盖、限额的意思。跟普通的集合相比,数据在写入这种集合时遵循FIFO原则。可以将这种集合想象为一个环状的队列,新文档在写入时会被插入队列的末尾,如果队列已满,那么之前的文档会被写入的文档覆盖。通过固定集合的大小,我们可以保证数据库只会存储“限额”的数据,超过该限额的旧数据都会被丢弃。

创建固定集合

db.createCollection("logs",{capped:true,size:4096,max:10})

max:指集合的文档数量最大值,这里是10条

size:指集合的空间占用最大值,这里是4096字节(4KB)

这两个参数会同时对集合的上限产生影响,也就是说,只要任一条件达到阈值都会被认为集合已经写满。其中size是必选的,而max则是可选的。

可以使用collection.stats命令查看文档的占用空间。

db.logs.stats()

测试:

尝试在这个集合中插入15条数据,再查询会发现,由于文档数量上限被设定为10条,前面插入的5条数据已经被覆盖。

for(var i=0;i<15;i++){db.logs.insert({t:"row-"+i})}

优势与限制

优势:

固定集合在底层使用的是顺序I/O操作,而普通集合使用的是随机I/O.顺序I/O在磁盘操作上由于寻道次数少而比随机I/O要高效得多,因此固定集合的写入性能是很高的。此外,如果按写入顺序进行数据读取,也会获得非常好的性能表现。

限制:

1、无法动态修改存储的上限,如果需要修改max或size,则只能先执行collection.drop命令,将集合删除后再重新创建。

2、无法删除已有数据,对固定集合中的数据进行删除将会得到如下错误。

3、对已有数据进行修改,新文档大小必须与原来的文档大小一致,否则不允许更新

4、默认情况下,固定集合只有一个_id索引,而且最好是按数据写入的顺序进行读取。也可以添加新的索引,但这会降低数据写入的性能

5、固定集合不支持分片,同时,在MongoDB 4.2版本中规定了事务中也无法对固定集合执行写操作。

适用场景

固定集合很适合用来存储一些“临时态”的数据。“临时态”意味着数据在一定程度上可以被丢弃。同时,用户还应该更关注最新的数据,随着时间的推移,数据的重要性逐渐降低,直至被淘汰处理。

适用场景如下:

1、系统日志,符合固定集合的特征,而日志系统通常也只需要一个固定的空间来存放日志。在MongoDB内部,副本集的同步日志(oplog)就使用了固定集合。

2、存储少量文档,如最新发布的TopN条文章信息。得益于内部缓存的作用,对于这种少量文档的查询是非常高效的。

使用固定集合实现FIFO队列

在股票实时系统中,大家最关心股票价格的变动。而应用系统中也需要根据这些实时的变化数据来分析当前的行情。倘若将股票的价格变化看作是一个事件,而股票交易所则是价格变动事件的“发布者”,股票APP、应用系统则是事件的“消费者”。这样,我们就可以将股票价格的发布、通知抽象为一种数据的消费行为,此时往往需要一个消息队列来实现该需求。

结合业务场景: 利用固定集合实现存储股票价格变动信息的消息队列

1、创建stock_queue消息队列,其可以容纳10MB的数据

db.createCollection("stock_queue",{capped:true,size:10485760})

2、定义消息格式

{timestamped: new Date(),stock: "MongoDB Inc",price: 20.33}

1、timestamp:指股票动态消息的产生时间

2、stock:指股票的名称

3、price:指股票的价格,是一个Double类型的字段

为了能支持按时间条件进行快速的检索,比如查询某个时间点之后的数据,可以为timestamp添加索引

db.stock_queue.createIndex({timestamped:1})

3、构建生产者,发布股票动态

模拟股票的实时变动

function pushEvent(){while(true){db.stock_queue.inset({timestamped:new Date(),stock: "MongoDB Inc",price: 100*Math.random(1000)})}print("publish stock changed");sleep(1000);}

执行pushEvent函数,此时客户端会每隔1秒向stock_queue中写入一条股票信息

pushEvent()

4、构建消费者,监听股票动态

对于消费方来说,更关心的是最新数据,同时还应该保持持续进行“拉取”,以便知晓实时发生的变化。根据这样的逻辑,可实现一个listen函数

function listen(){var cursor = db.stock_queue.find({timestamped:{$gte:new Date()}}).tailable();while(true){if(cursor.hasNext()){print(JSON.stringify(cursor.next(),null,2))}sleep(1000)}}

find操作的查询条件被指定为仅查询比当前时间更新的数据,而由于采用了读取游标的方式,因此游标在获取不到数据时并不会被关闭,这种行为非常类似于Linux中的tail-f命令。在一个循环中会定时检查是否有新的数据产生,一旦发现新的数据(cursor.hasNext()=true),则直接将数据打印到控制台。

执行这个监听函数,就可以看到实时发布的股票信息

listen()

本内容不代表本网观点和政治立场,如有侵犯你的权益请联系我们处理。
网友评论
网友评论仅供其表达个人看法,并不表明网站立场。