这章我们学习MongoDB的查询操作。

Introduction to find

find方法用于执行MongoDB的查询操作。它返回collecion中的documents子集,没有添加参数的话它将返回整个collection数据。

例:查找c的所有数据

db.c.find()

find的第一个参数是查询条件,决定了返回哪些documents数据。

例:我们想要查找用户表中年龄为27的用户

db.users.find({"age":27})

假如我们想要多条件查询怎么办?只要在第一个参数里继续添加条件即可:

db.users.find({"age":27,"username":"joe"})

Specifying Which Keys to Return

有些时候查询你不需要返回所有的键值对。这时你可以在find或者findOne中的第二个参数里指定你想要返回的。

举个例子,你只要查询"username"和"email",这时可以这样写:

db.users.find({},{"username":1,"email":1})

这时返回:

{
"_id":ObjectId("4XXXX"),
"username":"joe",
"email":"joe@example.com"
}

"_id"是默认返回的,如果你连它也不需要返回,只需要这样写:

db.users.find({},{"username":1,"email":1,"_id":0})

Query Conditionals

"$lt","$lte","$gt","$gte"用于比较操作,相当于<,<=,>和>=

例如你想要查找年龄在18到30岁之间的用户:

db.users.find({"age":{"$gte":18,"$lte":30}})

"$ne" 用于不等于操作,例如你想要查询名字不为"joe"的用户,那你可以

db.users.find({"username":{"$ne":"joe"}})

OR Queries

在MongoDB中OR有两种方法:"$in"和"$or"。

如果你一个单独的key值有超过1种可能的值需要匹配,这时我们会用到"$in"。

假设我们想要查询彩票号码为725或者542或者390的document,我们可以这样

db.raffle.find({"ticket_no":{"$in":[725,542,390]}})

"$in"方法是非常灵活的,它允许你指定的值是不同的类型。

"$in"的方向操作是"$nin",例如我们不想查询彩票号码为725,542的document:

db.raffle.find({"ticket_no":{"$nin":[725,542]}})

那么,如果我们想要查询"ticket_no"为725或者"winner"为true的document,应该怎么写呢?

db.raffle.find({"$or":[{"ticket_no":725},{"winner":true}]})

再来个复杂点的:

db.raffle.find({"$or":[{"ticket_no":{"$in":[725,542]}},{"winner":true}]})

聪明如你,一定知道上面的意思啦~

$not

"$not"从字面意思来看,我想大家都能猜的出来。注意 它可以应用于所有条件之上。

我们先来看下"$mod","$mod"是用来取被除数的操作,它的第一个参数是除数,第二个是余数。

db.users.find({"id_num":{"$mod":[5,1]}})

这个操作返回了用户的"id_num"为1,6,11,16...的document。

那么如果我们要求返回的"id_num"为2,3,4,5,7,8,9,10,12这些值该怎么办呢?这时候"$not"的作用就凸显出来了,我们只需要这样:

db.users.find({"id_num":{"$not":{"$mod":[5,1]}}})

"$not"在与正则表达式的结合使用中是非常有用的,以后我们会介绍。

Conditional Semantics

假如你看过上一篇介绍的update modifiers,你会注意到以$为前缀的key会在不同的位置。在查询中,"$lt"在document里面;在更新中,“$inc”在document外面,

一般情况下这句话是对的:conditionals是一个docment的内部key,modifiers是document的外部key。

多个查询条件可以在一个单独的key中。举例,你想要查询年龄在20到30之间的用户,你会在"age"这个查询key中用到"$gt"和"$lt":

db.users.find({"age":{"$gt":20,"$lt":30}})

多个update modifiers不能被用在同一个key上,例如这样是错误的:

{"$inc":{"age":1},"$set":{"age":40}}

因为它改变了age两次。

还有一些元操作是用在document的外部的:"$and","$or","$nor"

他们都和下面的例子用法相似:

db.users.find({"$and":[{"x":{"$lt":1}},{"x":4}]})

Type-Specific Queries

null

null比较奇怪,它不仅能匹配自己也能匹配"does not exist"(也就是指缺少那个键的),如果在c中有这么几条条数据

{"_id":ObjectId("4XXX"),"y":null}
{"_id":ObjectId("4XXX"),"y":1}
{"_id":ObjectId("4XXX"),"y":2}

这时我们输入查询语句:

db.c.find({"z":null})

发现返回数据为:

{"_id":ObjectId("4XXX"),"y":null}
{"_id":ObjectId("4XXX"),"y":1}
{"_id":ObjectId("4XXX"),"y":2}

那么我们只想找值为null,我们可以这样写:

db.c.find({"y":{"$in":[null],"$exists":true}})

Regular Expressions

正则表达式对于字符串匹配是非常有用的。例如我们想要查找用户名为Joe或者joe的用户:

db.users.find({"name":/joe/i})

正则表达式也可以匹配自己:

>db.foo.insert({"bar": /baz/})
>db.foo.find({"bar": /baz/})
{
"_id":ObjectId("4XXX"),
"bar": /baz/
}

Querying Arrays

$all

我们先来插入几条数据

db.food.insert({"_id":1,"fruit":["apple","banana","peach"]})
db.food.insert({"_id":2,"fruit":["apple","kumquat","orange"]})
db.food.insert({"_id":3,"fruit":["cherry","banana","apple"]})

这时我们可以用$all查询包含apple和banana的数据:

db.food.find({"fruit":{"$all":["apple","banana"]}})

查询结果:

db.food.insert({"_id":1,"fruit":["apple","banana","peach"]})
db.food.insert({"_id":3,"fruit":["cherry","banana","apple"]})

我们来看看第二条返回结果,"banana"在"apple"前面,而查询条件中的"banana"是在后面,由此我们可以看出数组排序是无关紧要的。

另外,当$all中的数组中只有一个元素时,它的用法相当于不使用$all。

例:{"fruit":{"$all":["apple"]}}用法等同与{"fruit":"apple"}

你也可以对数组中的特定索引值进行匹配,例:

db.food.find({"fruit.2":"peach"})

上面的意思是对fruit字段的第3个值进行匹配。

$size

"$size"是一个很有用的查询条件,它允许你根据数组的长度进行匹配查找。

db.food.find({"fruit":{"$size":3}})

"$size"不能结合别的条件一起使用(例如$gt)

db.food.find({"$size":{"$gt":3}})

上面这种用法是错误的。

$slice

在之前的章节我们已经说过find的第二个参数用于返回特殊的key值。"$slice"可以返回数组key值的子集。

例:

 db.food.find({},{"fruit":{"$slice":2}})

返回的值为:

{ "_id" : 1, "fruit" : [ "apple", "banana" ] }
{ "_id" : 2, "fruit" : [ "apple", "kumquat" ] }
{ "_id" : 3, "fruit" : [ "cherry", "banana" ] }

另外"$slice"还能做分页

db.food.find({},{"fruit":{"$slice":[2,2]}})

上面的2代表skip2个元素,总的意思就是取3-4的数据。

这章暂时介绍到这,下章我们继续学习MongoDB的查询操作。

MongoDB 学习三的更多相关文章

  1. mongodb学习(三) 安装和基本CRUD

    菜鸟啊...先吐槽一下自己 发现mongodb已经升级到2.6标准版了.  服务端最新安装方法: http://www.cnblogs.com/lzrabbit/p/3682510.html 一 准备 ...

  2. MongoDB学习(三)

    MongoDB条件操作符 $gt  > 大于 $lt   < 小于 $gte >= 大于等于 $lte  <= 小于等于 $ne  !=  不等于 条件操作符可用于查询语句中, ...

  3. MongoDb学习三(spring-data-mongodb)

    本文采用2个种配置方式.xml配置 代码配置方式进行数据库的连接.实现简单的增删该查等一些操作.代码都有注释官方文档如下https://docs.spring.io/spring-data/mongo ...

  4. MongoDB学习笔记(三)--权限 && 导出导入备份恢复 && fsync和锁

    权限                                                                                             绑定内网I ...

  5. Mongodb学习笔记一(Mongodb环境配置)

    Mongodb学习 说明: MongoDB由databases组成,database由collections组成,collection由documents组成,document由fileds组成.Mo ...

  6. MongoDB学习记录

    一.操作符 "$lt" :"<""$lte" :"<=""$gt" :"> ...

  7. MongoDB学习(四)客户端工具备份数据库

    在上一篇MongoDB学习(三)中讲解了如何在服务器端进行数据的导入导出与备份恢复,本篇介绍下如何利用客户端工具来进行远程服务器的数据备份到本地. 以客户端工具MongoVUE为例来进行讲解: 1.首 ...

  8. MongoDB学习之--安全和认证

    MongoDB学习之--安全和认证 本文主要介绍两部分内容,Mongodb的安全检查配置以及安全认证操作: 虽然确保系统安全是系统管理员的重要工作,但是作为程序员了解其机制也是大有好处的,毕竟不是每个 ...

  9. MongoDB 学习笔记(原创)

    MongoDB 学习笔记 mongodb 数据库 nosql 一.数据库的基本概念及操作 SQL术语/概念 MongoDB术语/概念 解释/说明 database database 数据库 table ...

随机推荐

  1. netsh配置Windows防火墙(advfirewall)

    有人可能会说,Windows防火墙有非常友好的用户界面,为什么要使用命令行界面来配置一个Windows防火墙?有 个人认为有一下原因(撇开有的人喜欢命令行不喜欢界面的 , o(∩_∩)o 哈哈) Fi ...

  2. svn不是内部或外部命令?

    svn不是内部或外部命令? 我的系统是Win7, [计算机]-->右键[属性]-->[高级系统设置]-->[环境变量]-->[系统变量 (S)]-->[Path]--&g ...

  3. C++: 多态 虚函数

    一.多态: 1.多态是什么:具有不同功能的函数可以用同一个函数名 2.静态多态:程序编译时决定,通过函数重载实现. 3.动态多态:程序运行时决定,通过虚函数实现. 二.虚函数: 1.引入目的:可以通过 ...

  4. linux环境设置export

    一. shell显示与设置环境变量 1.export //echo $PATH 2.export | grep ROS 3.export ROS_IP=192.168.0.5(添加环境变量ROS_IP ...

  5. [CQOI2018] 社交网络

    题目背景 当今社会,在社交网络上看朋友的消息已经成为许多人生活的一部分.通常,一个用户在社交网络上发布一条消息(例如微博.状态.Tweet等) 后,他的好友们也可以看见这条消息,并可能转发.转发的消息 ...

  6. 数据结构------------------二叉查找树(BST)的java实现

    数据结构------------------二叉查找树(BST)的java实现 二叉查找树(BST)是一种能够将链表插入的灵活性和有序数组查找的高效性相结合的一种数据结构.它的定义如下: 二叉查找树是 ...

  7. Java加密技术(八)——数字证书

    原文:http://snowolf.iteye.com/blog/391931 请大家在阅读本篇内容时先阅读 Java加密技术(四),预先了解RSA加密算法. 在构建Java代码实现前,我们需要完成证 ...

  8. [IOS笔记] - 动画animation

    //移动 - (IBAction)translation:(id)sender { CABasicAnimation *traslation = [CABasicAnimation animation ...

  9. 【JVM】idea启动项目时候 添加jvm启动参数显示详细日志

    -verbose:class

  10. dubbo开发者指南

    开发者指南 参与 流程 任务 版本管理 源码构建 框架设计 整体设计 模块分包 依赖关系 调用链 暴露服务时序 引用服务时序 领域模型 基本原则 扩展点加载 扩展点配置 扩展点自动包装 扩展点自动装配 ...