MongoDB实战指南(二):索引与查询优化

数据库保存记录的机制是建立在文件系统上的,索引也是以文件的形式存储在磁盘上,在数据库中用到最多的索引结构就是B树。尽管索引在数据库领域是不可缺少的,但是对一个表建立过多的索引会带来一些问题,索引的建立要花费系统时间,同时索引文件也会占用磁盘空间。如果并发写入的量很大,每个插入的文档都要建立索引,可想而知,性能会较低。因此合理的建立索引是关键,搞清楚哪些字段上面需要建立索引,索引以什么样的方式建立,我们需要对每个查询过程进行分析,才能得出合理的结论。

1. 索引

在MongoDB上面,索引能够提高读操作及查询性能。没有索引,Mongodb必须扫描集合中的每一个文档,然后选择与查询条件匹配的文档,这种全表扫描的方式是非常低效的。MongoDB索引的数据结构也是B+树,它能够存储一小部分集合的数据,集体来说就是存储集合中建有索引的一个或多个字段的值,而且按照值的升序或者降序排列。对于一个查询来说,如果存在合适的索引,MongoDB能够利用这个索引减少文档的扫描数量,甚至对于某些查询能够直接从索引中返回结果,不需要再去扫描数据集合,这种查询是非常高效的。

1.1 单字段索引

MongoDB默认为所有集合都建立了一个_id字段的单字段索引,而且这个索引是唯一的,不能被删除,_id字段作为一个集合的主键,值是唯一的,对于一个集合来说,也可以在其他字段上创建单字段的唯一索引,如下面所述。

先插入一些数据:

> for(var i=1;i<10;i++){db.customers.insert({"name":"jordan"+i,"country":"American"})}
> for(var i=1;i<10;i++){db.customers.insert({"name":"gaga"+i,"country":"American"})}
> for(var i=1;i<10;i++){db.customers.insert({"name":"ham"+i,"country":"UK"})}
> for(var i=1;i<10;i++){db.customers.insert({"name":"brown"+i,"country":"UK"})}
> for(var i=1;i<10;i++){db.customers.insert({"name":"ramda"+i,"country":"Malaysia"})}

下面建立单字段唯一索引

> db.customers.ensureIndex({name:1},{unique:true})

单字段唯一索引去掉{unique:true}选项就是一个普通的单字段索引。

唯一索引创建成功后,会在相应数据库的系统集合system.indexs中增加一条索引记录,如下所示:

> db.system.indexes.find()

最后一条画红线的是刚刚添加的唯一索引。索引记录中v表示索引的版本;key表示索引建立在哪个字段上;1表示索引按照升序排列;索引记录所在的命名空间,name表示唯一索引的名称。唯一索引与普通索引的区别是要求插入的所有记录在创建索引的键值上唯一。

下面执行查询,一个用索引字段作为查询选择器;一个不用索引字段作为查询选择器进行比较。

> db.customers.find({"name":"ramda9"}).explain()

以上查询语句执行返回的结果用到了刚刚创建的索引:

> db.customers.find({"country":"Malaysia"}).explain()

以上查询语句执行返回的结果没有用到刚刚创建的索引。

1.2 复合索引

MongoDB支持多个字段的复合索引,复合索引支持匹配多个字段的查询。

给上面插入的数据创建一个复合索引:

> db.customers.ensureIndex({name:1,country:1})

1.3 数组的多键索引

如果对一个值为数组类型的字段创建索引,则会默认对数组中的每一个元素创建索引。

1.4 索引管理

通过上面创建的索引可以看到,索引记录都保存在特殊的集合system.indexs中。创建索引的语法如下所示:

>db.collection.ensureIndex(keys,options)

keys是一个document文档,包含需要添加索引的字段和索引的排序方向;option是可选参数,控制索引的创建方式。

索引的删除并不是直接找到索引所在的集合system.indexs,通过在集合上执行remove命令来删除,而是通过执行集合上的命令dropIndex来删除的。例如删除上面创建的如下复合索引。

> db.customers.dropIndex("name_1_country_1")

其中参数为索引的名称。

2. 查询优化

  查询优化的目的是找出慢的查询语句,分析慢的原因,然后优化此查询语句。

  Mongodb对于超过100ms的查询语句,会自动地输出到日志文件里面,因此找出慢查询的第一步是查看MongoDB的日志文件,如果觉得这100ms阈值过大,可以通过mongod的服务启动选项showms来设置,它的默认值是100ms。

  用上面的方法找出慢查询可能比较粗糙,第二种定位慢查询的方法是打开数据库的监视功能,它默认是关闭的,可以通过下面的命令打开。

  db.setProfilingLevel(level,[slowms])

  参数level是监视级别,值为0表示关闭数据库的监视功能,为1表示只记录慢查询,为2表示记录所有操作;slown为可选参数,设定慢查询的阈值。

  所有监视的结果都将保存到一个特殊的集合systemn.profile中。

  通过上面的两种方法可以找出慢查询的语句,然后通过建立相应的索引基本可以解决绝大部分的问题。但是有时我们需要更加精细的优化代码,这就需要分析这些慢查询的执行计划,查看查询是否用到索引,是否与我们想要的执行计划相同,用Mongodb的explian命令可以查看执行计划。

转载于:https://www.cnblogs.com/longshiyVip/p/5095935.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/357819.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

mysql学生选课系统的关系模型_使用PowerDesigner搭建学生选课管理系统(学生老师管理员一体系结构)由基础设计至数据库生成(SQL语句源代码的生成)全过程实例操作...

思考构图大小及范围首先&#xff0c;我们大致的写出要建的几张表&#xff0c;并自定义其Code(所谓Code其实就是自定义一个字段名&#xff0c;因为SQL语句不能由中文)我们可以将不同的表按照不同的类型分开&#xff0c;这样一个类型的表可以一起建并且相互之间的关系不会弄混制作…

java实现飞机大战游戏

导读:现如今,随着智能手机的兴起与普及,加上5G(the 5th Generation mobile communication ,第5代移动通信技术)网络的深入,越来越多的IT行业开始向手机行业转移重心。而手机行业中游戏方面的利润所占比重较大,并且手机游戏大多数则是由Java语言开发研制的。所以我想顺应…

merge语句使用_使用SQL:2003 MERGE语句的奥术魔术

merge语句使用时不时地&#xff0c;由于以下任何原因&#xff0c;我们不得不将INSERT与UPDATE区分开来感到尴尬&#xff1a; 我们必须至少发表两个声明 我们必须考虑性能 我们必须考虑比赛条件 我们必须在[UPDATE; 如果UPDATE_COUNT 0 THEN INSERT]和[INSERT; 如果例外然后…

UVA 10917 Walk Through the Forest

最短路DP Walk Through the ForestTime Limit: 3000MS Memory Limit: Unknown 64bit IO Format: %lld & %llu[Submit] [Go Back] [Status] Description Problem C: A Walk Through the Forest Jimmy experiences a lot of stress at work these days, especially sin…

一文教你用java实现愤怒的小鸟游戏

导读&#xff1a;Java语言是一门面向对象的编程语言&#xff0c;它不但汲取了C语言的各种精髓&#xff0c;而且还抛弃了C语言里晦涩难懂的多继承和指针等概念&#xff0c;所以Java语言具有的特征便是&#xff1a;功能超强和易用实用。Jbox2D中不仅集成了大量物理运动学和机械运…

WinForm - 两个窗体之间的方法调用

方法1&#xff1a; 所有权法//Form1://需要有一个公共的刷新方法public void Refresh_Method(){  //...} //在调用Form2时,要把Form2的所有者设为Form1Form2 f2 new Form2() ;f2.Owner this;f2.ShowDialog() ; //Form2://在需要对其调用者&#xff0…

一文教你用java实现儿时的超级玛丽游戏

导读&#xff1a;近年来&#xff0c;Java作为一种新的编程语言&#xff0c;以其简单性、可移植性和平台无关性等优点&#xff0c;得到了广泛地应用。J2SE称为Java标准版或Java标准平台。J2SE提供了标准的SDK开发平台。利用该平台可以开发Java桌面应用程序和低端的服务器应用程序…

java开发一款雷电游戏

导读:电脑游戏,是指在计算机上能够运转的游戏软件。这种软件具有较强的娱乐性。电脑游戏的创新和发展与硬件、软件的发展紧密相关。它能够给玩家提供一个虚拟的环境,使游戏带给了人们很多的享受和欢乐。雷电游戏因为操作简单,节奏明快,一直是纵轴射击游戏的经典之作。经常…

glassfish_具有GlassFish和一致性的高性能JPA –第2部分

glassfish在我的四部分系列的第二部分中&#xff0c;我将解释将Coherence与EclipseLink和GlassFish结合使用的第一个策略。它描述了配置带有Coherence的JPA支持的Cache所必须采取的步骤&#xff0c;以及如何在GlassFish中使用它。高性能数据存储。 一般的做法 您可以将Coherenc…

java开发一个连连看小游戏

导读:连连看是一种消除类益智游戏,核心要求是在规定的时间内,消除游戏界面中选中的两张相同的图案,直至完全消除所有图案。这款游戏操作简单(只需单击鼠标左键操作)、面向人群广泛,在限时操作的游戏要求下,吸引玩家自发地锻炼观察能力、判断能力和反应能力,故从推出至今…

java枚举的特点_Java中有些好的特性(二):枚举

前言我在写上一篇时&#xff0c;开始选了一个很土的名字“Java超过C#的地方”&#xff0c;然后引起了某些同学的不满&#xff0c;后来修改了个名。我在这里再次申明一下&#xff0c;我没有贬低任何语言之意&#xff0c;纯粹是从一个.NET程序员的角度去看Java&#xff0c;看看是…

java开发一款推箱子游戏

导读:社会在进步,人们生活质量也在日益提高。高强度的压力也接踵而来。社会中急需出现新的有效方式来缓解人们的压力。此次设计符合了社会需求,Java推箱子游戏可以让人们在闲暇之余,体验游戏的乐趣。具有操作简单,易于上手的特点。 推箱子游戏的玩法十分简单——控制人物绕…

java ee技术_Java EE 6与Spring Framework:技术决策过程

java ee技术在过去的几个月中&#xff0c;我们经历了这个决策过程&#xff1a;为Java平台上的企业开发选择哪种技术堆栈&#xff1f; 有多种选择。 但是&#xff0c;我们深入讨论的是&#xff1a;纯Java EE 6堆栈与带有Java EE的Spring。 以下博客文章总结了当您考虑这些技术堆…

java开发一款坦克大战游戏

导读:随着人们对生活质量的要求一天比一天高,为了让人们更好地开掘自身的智慧,游戏就此进入了大众的视野,在人们的生活中有着重要的位置,已然变得必不可少。游戏产业推动高新技术不断升级,极大地促进了经济的增长,推动了“第四产业”的经济腾飞。坦克大战游戏是童年时期…

java大转盘抽奖概率算法_幸运大转盘抽奖 抽奖算法 程序实现逻辑

近期碰到的一个需求&#xff0c;实现一个类似大转盘抽奖的功能&#xff0c;需自定义奖项&#xff0c;各奖项中奖概率&#xff0c;当日抽奖最大次数&#xff0c;抽奖成本等。分享一个简单的java代码的实现的思路&#xff0c;有不足之处感谢各位指正。初步方法首先要定义几个奖品…

java开发银行柜员业务绩效考核系统

导读:当今社会己进入信息社会时代,信息己经受到社会的广泛关注,被看作社会和科学技术发展的三大支柱(材料、能源、信息)之一。信息是管理的基础,是进行决策的的基本依据。在一个组织里,信息己作为人力、物力、财力之外的第四种能源,占有重要的地位。然而,信息是一种非…

一文教你使用java开发一款坦克大战游戏

导读&#xff1a;随着人们对生活质量的要求一天比一天高&#xff0c;为了让人们更好地开掘自身的智慧&#xff0c;游戏就此进入了大众的视野&#xff0c;在人们的生活中有着重要的位置&#xff0c;已然变得必不可少。游戏产业推动高新技术不断升级&#xff0c;极大地促进了经济…

Java 8流中的常见SQL子句及其等效项

功能编程允许使用通用语言进行准声明性编程 。 通过使用功能强大的流畅API&#xff08;例如Java 8的Stream API &#xff09;或jOOλ的顺序Stream扩展Seq或更复杂的库&#xff08;例如javaslang或functionaljava&#xff09; &#xff0c;我们可以以一种非常简洁的方式来表示数…

混合使用Azure LB和ILB访问相同web服务(3)

接下来我们来配置Azure Load balancer&#xff0c;就是面向公网的负载均衡器&#xff1a; 1.在该测试中&#xff0c;为了保持内网访问和外网访问一样的体验&#xff0c;本地端口和public端口和ILB一样&#xff0c;同样是80&#xff1a; PS C:\> Get-AzureVM -ServiceName …

mysql日活统计函数_如何通过简化日活模型,预估一个产品的日活(DAU)?

本文笔者通过简化日活模型&#xff0c;粗略地估算出未来一段时间的产品日活规模。进而通过日活的规模&#xff0c;再去估算一些潜在收益&#xff0c;以及运营成本等等数据。这段时间&#xff0c;经常有人问到以下的问题&#xff1a;按照现在的推广和留存&#xff0c;我们在未来…