摘要:上一篇文章指南更新文檔下一篇文章指南特定類(lèi)型的查詢(xún)本章將詳細(xì)介紹查詢(xún)。查詢(xún)條件和就是全部的比較操作符,分別對(duì)應(yīng)和。如果查詢(xún)優(yōu)化器可以更高效地處理,那就選擇使用它。注意,查詢(xún)優(yōu)化器不會(huì)對(duì)進(jìn)行優(yōu)化,這與其他操作符不同。
上一篇文章:MongoDB指南---6、更新文檔
下一篇文章:MongoDB指南---8、特定類(lèi)型的查詢(xún)
本章將詳細(xì)介紹查詢(xún)。主要會(huì)涵蓋以下幾個(gè)方面:
使用find或者findOne函數(shù)和查詢(xún)文檔對(duì)數(shù)據(jù)庫(kù)執(zhí)行查詢(xún);
使用$條件查詢(xún)實(shí)現(xiàn)范圍查詢(xún)、數(shù)據(jù)集包含查詢(xún)、不等式查詢(xún),以及其他一些查詢(xún);
查詢(xún)將會(huì)返回一個(gè)數(shù)據(jù)庫(kù)游標(biāo),游標(biāo)只會(huì)在你需要時(shí)才將需要的文檔批量返回;
還有很多針對(duì)游標(biāo)執(zhí)行的元操作,包括忽略一定數(shù)量的結(jié)果,或者限定返回結(jié)果的數(shù)量,以及對(duì)結(jié)果排序。
4.1 find簡(jiǎn)介MongoDB中使用find來(lái)進(jìn)行查詢(xún)。查詢(xún)就是返回一個(gè)集合中文檔的子集,子集合的范圍從0個(gè)文檔到整個(gè)集合。find的第一個(gè)參數(shù)決定了要返回哪些文檔,這個(gè)參數(shù)是一個(gè)文檔,用于指定查詢(xún)條件。
空的查詢(xún)文檔(例如{})會(huì)匹配集合的全部?jī)?nèi)容。要是不指定查詢(xún)文檔,默認(rèn)就是{}。例如:
> db.c.find()
將批量返回集合c中的所有文檔。
開(kāi)始向查詢(xún)文檔中添加鍵/值對(duì)時(shí),就意味著限定了查詢(xún)條件。對(duì)于絕大多數(shù)類(lèi)型來(lái)說(shuō),這種方式很簡(jiǎn)單明了。數(shù)值匹配數(shù)值,布爾類(lèi)型匹配布爾類(lèi)型,字符串匹配字符串。查詢(xún)簡(jiǎn)單的類(lèi)型,只要指定想要查找的值就好了,十分簡(jiǎn)單。例如,想要查找"age"值為27的所有文檔,直接將這樣的鍵/值對(duì)寫(xiě)進(jìn)查詢(xún)文檔就好了:
> db.users.find({"age" : 27})
要是想匹配一個(gè)字符串,比如值為"joe"的"username"鍵,那么直接將鍵/值對(duì)寫(xiě)在查詢(xún)文檔中即可:
> db.users.find({"username" : "joe"})
可以向查詢(xún)文檔加入多個(gè)鍵/值對(duì),將多個(gè)查詢(xún)條件組合在一起,這樣的查詢(xún)條件會(huì)被解釋成“條件1AND條件2AND ... AND條件N”。例如,要想查詢(xún)所有用戶(hù)名為joe且年齡為27歲的用戶(hù),可以像下面這樣:
> db.users.find({"username" : "joe", "age" : 27})4.1.1 指定需要返回的鍵
有時(shí)并不需要將文檔中所有鍵/值對(duì)都返回。遇到這種情況,可以通過(guò)find(或者findOne)的第二個(gè)參數(shù)來(lái)指定想要的鍵。這樣做既會(huì)節(jié)省傳輸?shù)臄?shù)據(jù)量,又能節(jié)省客戶(hù)端解碼文檔的時(shí)間和內(nèi)存消耗。
例如,如果只對(duì)用戶(hù)集合的"username"和"email"鍵感興趣,可以使用如下查詢(xún)返回這些鍵:
> db.users.find({}, {"username" : 1, "email" : 1}) { "_id" : ObjectId("4ba0f0dfd22aa494fd523620"), "username" : "joe", "email" : "joe@example.com" }
可以看到,默認(rèn)情況下"_id"這個(gè)鍵總是被返回,即便是沒(méi)有指定要返回這個(gè)鍵。
也可以用第二個(gè)參數(shù)來(lái)剔除查詢(xún)結(jié)果中的某些鍵/值對(duì)。例如,文檔中有很多鍵,但是我們不希望結(jié)果中含有"fatal_weakness"鍵:
> db.users.find({}, {"fatal_weakness" : 0})
使用這種方式,也可以把"_id"鍵剔除掉:
> db.users.find({}, {"username" : 1, "_id" : 0}) { "username" : "joe", }4.1.2 限制
查詢(xún)的使用上有些限制。傳遞給數(shù)據(jù)庫(kù)的查詢(xún)文檔的值必須是常量。(在你自己的代碼里可以是正常的變量。)也就是不能引用文檔中其他鍵的值。例如,要想保持庫(kù)存,有"in_stock"(剩余庫(kù)存)和"num_sold"(已出售)兩個(gè)鍵,想通過(guò)下列查詢(xún)來(lái)比較兩者的值是行不通的:
> db.stock.find({"in_stock" : "this.num_sold"}) // 這樣是行不通的
的確有辦法實(shí)現(xiàn)類(lèi)似的操作(詳見(jiàn)4.4節(jié)),但通常需要略微修改一下文檔結(jié)構(gòu),就能通過(guò)普通查詢(xún)來(lái)完成這樣的操作了,這種方式性能更好。在這個(gè)例子中,可以在文檔中使用"initial_stock"(初始庫(kù)存)和"in_stock"兩個(gè)鍵。這樣,每當(dāng)有人購(gòu)買(mǎi)物品,就將"in_stock"減去1。這樣,只需要用一個(gè)簡(jiǎn)單的查詢(xún)就能知道哪種商品已脫銷(xiāo):
> db.stock.find({"in_stock" : 0})4.2 查詢(xún)條件
查詢(xún)不僅能像前面說(shuō)的那樣精確匹配,還能匹配更加復(fù)雜的條件,比如范圍、OR子句和取反。
4.2.1 查詢(xún)條件"$lt"、"$lte"、"$gt"和"$gte"就是全部的比較操作符,分別對(duì)應(yīng)<、<=、>和>=。可以將其組合起來(lái)以便查找一個(gè)范圍的值。例如,查詢(xún)18~30歲(含)的用戶(hù),就可以像下面這樣:
> db.users.find({"age" : {"$gte" : 18, "$lte" : 30}})
這樣就可以查找到"age"字段大于等于18、小于等于30的所有文檔。
這樣的范圍查詢(xún)對(duì)日期尤為有用。例如,要查找在2007年1月1日前注冊(cè)的人,可以像下面這樣:
> start = new Date("01/01/2007") > db.users.find({"registered" : {"$lt" : start}})
可以對(duì)日期進(jìn)行精確匹配,但是用處不大,因?yàn)槲臋n中的日期是精確到毫秒的。而我們通常是想得到一天、一周或者是一個(gè)月的數(shù)據(jù),這樣的話(huà),使用范圍查詢(xún)就很有必要了。
對(duì)于文檔的鍵值不等于某個(gè)特定值的情況,就要使用另外一種條件操作符"$ne"了,它表示“不相等”。若是想要查詢(xún)所有名字不為joe的用戶(hù),可以像下面這樣查詢(xún):
> db.users.find({"username" : {"$ne" : "joe"}})
"$ne"能用于所有類(lèi)型的數(shù)據(jù)。
4.2.2 OR查詢(xún)MongoDB中有兩種方式進(jìn)行OR查詢(xún):"$in"可以用來(lái)查詢(xún)一個(gè)鍵的多個(gè)值;"$or"更通用一些,可以在多個(gè)鍵中查詢(xún)?nèi)我獾慕o定值。
如果一個(gè)鍵需要與多個(gè)值進(jìn)行匹配的話(huà),就要用"$in"操作符,再加一個(gè)條件數(shù)組。例如,抽獎(jiǎng)活動(dòng)的中獎(jiǎng)號(hào)碼是725、542和390。要找出全部的中獎(jiǎng)文檔的話(huà),可以構(gòu)建如下查詢(xún):
> db.raffle.find({"ticket_no" : {"$in" : [725, 542, 390]}})
"$in"非常靈活,可以指定不同類(lèi)型的條件和值。例如,在逐步將用戶(hù)的ID號(hào)遷移成用戶(hù)名的過(guò)程中,查詢(xún)時(shí)需要同時(shí)匹配ID和用戶(hù)名:
> db.users.find({"user_id" : {"$in" : [12345, "joe"]})
這會(huì)匹配"user_id"等于12345的文檔,也會(huì)匹配"user_id"等于"joe"的文檔。
要是"$in"對(duì)應(yīng)的數(shù)組只有一個(gè)值,那么和直接匹配這個(gè)值效果一樣。例如,{ticket_no : {$in:[725]}}和{ticket_no : 725}的效果一樣。
與"$in"相對(duì)的是"$nin","$nin"將返回與數(shù)組中所有條件都不匹配的文檔。要是想返回所有沒(méi)有中獎(jiǎng)的人,就可以用如下方法進(jìn)行查詢(xún):
> db.raffle.find({"ticket_no" : {"$nin" : [725, 542, 390]}})
該查詢(xún)會(huì)返回所有沒(méi)有中獎(jiǎng)的人。
"$in"能對(duì)單個(gè)鍵做OR查詢(xún),但要是想找到"ticket_no"為725或者"winner"為true的文檔該怎么辦呢?對(duì)于這種情況,應(yīng)該使用"$or"。"$or"接受一個(gè)包含所有可能條件的數(shù)組作為參數(shù)。上面中獎(jiǎng)的例子如果用"$or"改寫(xiě)將是下面這個(gè)樣子:
> db.raffle.find({"$or" : [{"ticket_no" : 725}, {"winner" : true}]})
"$or"可以包含其他條件。例如,如果希望匹配到中獎(jiǎng)的"ticket_no",或者"winner"鍵的值為true的文檔,就可以這么做:
> db.raffle.find({"$or" : [{"ticket_no" : {"$in" : [725, 542, 390]}}, {"winner" : true}]})
使用普通的AND型查詢(xún)時(shí),總是希望盡可能用最少的條件來(lái)限定結(jié)果的范圍。OR型查詢(xún)正相反:第一個(gè)條件應(yīng)該盡可能匹配更多的文檔,這樣才是最為高效的。
"$or"在任何情況下都會(huì)正常工作。如果查詢(xún)優(yōu)化器可以更高效地處理"$in",那就選擇使用它。
"$not"是元條件句,即可以用在任何其他條件之上。就拿取模運(yùn)算符"$mod"來(lái)說(shuō)。"$mod"會(huì)將查詢(xún)的值除以第一個(gè)給定值,若余數(shù)等于第二個(gè)給定值則匹配成功:
> db.users.find({"id_num" : {"$mod" : [5, 1]}})
上面的查詢(xún)會(huì)返回"id_num"值為1、6、11、16等的用戶(hù)。但要是想返回"id_num"為2、3、4、5、7、8、9、10、12等的用戶(hù),就要用"$not"了:
> db.users.find({"id_num" : {"$not" : {"$mod" : [5, 1]}}})
"$not"與正則表達(dá)式聯(lián)合使用時(shí)極為有用,用來(lái)查找那些與特定模式不匹配的文檔(4.3.2節(jié)會(huì)詳細(xì)講述正則表達(dá)式的使用)。
4.2.4 條件語(yǔ)義如果比較一下上一章的更新修改器和前面的查詢(xún)文檔,會(huì)發(fā)現(xiàn)以$開(kāi)頭的鍵位于在不同的位置。在查詢(xún)中,"$lt"在內(nèi)層文檔,而更新中"$inc"則是外層文檔的鍵。基本可以肯定:條件語(yǔ)句是內(nèi)層文檔的鍵,而修改器則是外層文檔的鍵。
可以對(duì)一個(gè)鍵應(yīng)用多個(gè)條件。例如,要查找年齡為20~30的所有用戶(hù),可以在"age"鍵上使用"$gt"和"$lt":
> db.users.find({"age" : {"$lt" : 30, "$gt" : 20}})
一個(gè)鍵可以有任意多個(gè)條件,但是一個(gè)鍵不能對(duì)應(yīng)多個(gè)更新修改器。例如,修改器文檔不能同時(shí)含有{"$inc" : {"age" : 1}, "$set" : {age : 40}},因?yàn)樾薷牧?age"兩次。但是對(duì)于查詢(xún)條件句就沒(méi)有這種限定。
有一些“元操作符”(meta-operator)也位于外層文檔中,比如"$and"、"$or"和"$nor"。它們的使用形式類(lèi)似:
> db.users.find({"$and" : [{"x" : {"$lt" : 1}}, {"x" : 4}]})
這個(gè)查詢(xún)會(huì)匹配那些"x"字段的值小于等于1并且等于4的文檔。雖然這兩個(gè)條件看起來(lái)是矛盾的,但是這是完全有可能的,比如,如果"x"字段的值是這樣一個(gè)數(shù)組{"x" : [0, 4]},那么這個(gè)文檔就與查詢(xún)條件相匹配。注意,查詢(xún)優(yōu)化器不會(huì)對(duì)"$and"進(jìn)行優(yōu)化,這與其他操作符不同。如果把上面的查詢(xún)改成下面這樣,效率會(huì)更高:
> db.users.find({"x" : {"$lt" : 1, "$in" : [4]}})
上一篇文章:MongoDB指南---6、更新文檔
下一篇文章:MongoDB指南---8、特定類(lèi)型的查詢(xún)
文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請(qǐng)注明本文地址:http://m.specialneedsforspecialkids.com/yun/19561.html
摘要:上一篇文章指南更新文檔下一篇文章指南特定類(lèi)型的查詢(xún)本章將詳細(xì)介紹查詢(xún)。查詢(xún)條件和就是全部的比較操作符,分別對(duì)應(yīng)和。如果查詢(xún)優(yōu)化器可以更高效地處理,那就選擇使用它。注意,查詢(xún)優(yōu)化器不會(huì)對(duì)進(jìn)行優(yōu)化,這與其他操作符不同。 上一篇文章:MongoDB指南---6、更新文檔下一篇文章:MongoDB指南---8、特定類(lèi)型的查詢(xún) 本章將詳細(xì)介紹查詢(xún)。主要會(huì)涵蓋以下幾個(gè)方面: 使用find或者f...
摘要:但有時(shí)我們希望返回與查詢(xún)條件相匹配的任意一個(gè)數(shù)組元素。首先,可以使用要求同時(shí)使用查詢(xún)條件中的兩個(gè)語(yǔ)句與一個(gè)數(shù)組元素進(jìn)行比較。 上一篇文章:MongoDB指南---7、find簡(jiǎn)介與查詢(xún)條件下一篇文章:MongoDB指南---9、游標(biāo)與數(shù)據(jù)庫(kù)命令 如第2章所述,MongoDB的文檔可以使用多種類(lèi)型的數(shù)據(jù)。其中有一些在查詢(xún)時(shí)會(huì)有特別的表現(xiàn)。 4.3.1 null null類(lèi)型的行為有點(diǎn)奇...
摘要:但有時(shí)我們希望返回與查詢(xún)條件相匹配的任意一個(gè)數(shù)組元素。首先,可以使用要求同時(shí)使用查詢(xún)條件中的兩個(gè)語(yǔ)句與一個(gè)數(shù)組元素進(jìn)行比較。 上一篇文章:MongoDB指南---7、find簡(jiǎn)介與查詢(xún)條件下一篇文章:MongoDB指南---9、游標(biāo)與數(shù)據(jù)庫(kù)命令 如第2章所述,MongoDB的文檔可以使用多種類(lèi)型的數(shù)據(jù)。其中有一些在查詢(xún)時(shí)會(huì)有特別的表現(xiàn)。 4.3.1 null null類(lèi)型的行為有點(diǎn)奇...
摘要:可以通過(guò)來(lái)強(qiáng)制使用某個(gè)特定的索引,再次執(zhí)行這個(gè)查詢(xún),但是這次使用,作為索引。 上一篇文章:MongoDB指南---9、游標(biāo)與數(shù)據(jù)庫(kù)命令下一篇文章:MongoDB指南---11、使用復(fù)合索引、$操作符如何使用索引、索引對(duì)象和數(shù)組、索引基數(shù) 本章介紹MongoDB的索引,索引可以用來(lái)優(yōu)化查詢(xún),而且在某些特定類(lèi)型的查詢(xún)中,索引是必不可少的。 什么是索引?為什么要用索引? 如何選擇需要建立...
摘要:可以通過(guò)來(lái)強(qiáng)制使用某個(gè)特定的索引,再次執(zhí)行這個(gè)查詢(xún),但是這次使用,作為索引。 上一篇文章:MongoDB指南---9、游標(biāo)與數(shù)據(jù)庫(kù)命令下一篇文章:MongoDB指南---11、使用復(fù)合索引、$操作符如何使用索引、索引對(duì)象和數(shù)組、索引基數(shù) 本章介紹MongoDB的索引,索引可以用來(lái)優(yōu)化查詢(xún),而且在某些特定類(lèi)型的查詢(xún)中,索引是必不可少的。 什么是索引?為什么要用索引? 如何選擇需要建立...
閱讀 1706·2021-11-24 09:39
閱讀 2489·2021-11-18 10:07
閱讀 3668·2021-08-31 09:40
閱讀 3341·2019-08-30 15:44
閱讀 2637·2019-08-30 12:50
閱讀 3658·2019-08-26 17:04
閱讀 1435·2019-08-26 13:49
閱讀 1271·2019-08-23 18:05