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…

python定义区间[-5、5_有一个数 x 在区间 [-5,0] 内 , 写出其条件表达式 。 (5.0分)_学小易找答案...

【其它】2018应心ABC反应时任务实验数据.sav 请利用上面的实验数据文档,完成本次实验的报告撰写,并上传。 【单选题】“ab”+”c”*2 结果是: (3.0分) 【判断题】元组可以作为集合的元素。 (2.0分) 【判断题】函数中必须包含return语句。 (3.5分) 【判断题】加法运算符可以用来…

java实现愤怒的小鸟游戏

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

为什么应始终将连接池与Oracle XE一起使用

介绍 Oracle Express Edition是Oracle Enterprise Edition的免费版本&#xff0c;其较小的尺寸使其非常方便地测试各种Oracle功能。 根据Oracle文档 &#xff0c;Express Edition最多可以使用一个CPU和1 GB RAM&#xff0c;但是实际上存在其他限制并不总是很明显。 数据库连接…

异想维度 java_Java实现多字段(维度)复杂排序

//Java 实现多字段排序HashMap map1 new HashMap();map1.put("dataindex0", null);map1.put("dataindex1", 6);map1.put("dataindex2", 1.1);map1.put("id", 1);HashMap map2 new HashMap();map2.put("dataindex0", "…

Linux下mysql的root密码忘记解决方法

方法一&#xff1a; 1&#xff0e;首先确认服务器出于安全的状态&#xff0c;也就是没有人能够任意地连接MySQL数据库。 因为在重新设置MySQL的root密码的期间&#xff0c;MySQL数据库完全出于没有密码保护的 状态下&#xff0c;其他的用户也可以任意地登录和修改MySQL的信息…

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

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

java中大数开方_大数开方(Java版)

Java代码&#xff1a;适合被开方数不超过1000位的。import java.util.*;import java.math.*;public class Main{public static void main(String[] args){Scanner cinnew Scanner(System.in);int tcin.nextInt();BigInteger remainBigInteger.ZERO;BigInteger oddBigInteger.ZE…

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桌面应用程序和低端的服务器应用程序…

'mysql_attr_use_buffered_query'_php中mysql操作的buffer知识

php与mysql的连接有三种方式&#xff0c;mysql&#xff0c;mysqli&#xff0c;pdo。不管使用哪种方式进行连接&#xff0c;都有使用buffer和不使用buffer的区别。什么叫使用buffer和不使用buffer呢&#xff1f;客户端与mysql服务端进行查询操作&#xff0c;查询操作的时候如果获…

java开发一款雷电游戏

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

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

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

java开发一个连连看小游戏

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

将java.util.concurrent.BlockingQueue用作rx.Observable

在Java中&#xff0c;经典的生产者&#xff0d;消费者模式相对简单&#xff0c;因为我们有java.util.concurrent.BlockingQueue 。 为了避免繁忙的等待和容易出错的手动锁定&#xff0c;我们只需利用put()和take() 。 如果队列已满或为空&#xff0c;它们都将阻塞。 我们需要的…

Mac OS X 10.10如何打开虚拟内存

转载请注明出处&#xff1a;http://www.cnblogs.com/blazer/p/5103608.html 禁用虚拟内存sudo launchctl unload -w /System/Library/LaunchDaemons/com.apple.metadata.mds.plist删除交换文件sudo rm /private/var/vm/swapfile*启用虚拟内存sudo launchctl load -w /System/L…

java 1.5.0 gcj_CentOS安装JAVA后JAVA版本不对的问题

今天用CentOS安装JDK&#xff0c;发觉在安装完成后&#xff0c;输入java命令来验证是否安装成功时&#xff0c;出现Usage: gij [OPTION] ... CLASS [ARGS] ...to invoke CLASS.main, orgij -jar [OPTION] ... JARFILE [ARGS] ...to execute a jar fileTry gij --help‘ for mor…