MongoDB 基本操作詳解

jopen 9年前發布 | 12K 次閱讀 MongoDB NoSQL數據庫

MongoDB 最大的特點是他支持的查詢語言非常強大,其語法有點類似于面向對象的查詢語 言,幾乎可以實現類似關系數據庫單表查詢的絕大部分功能,而且還支持對數據建立索引。最后由于 MongoDB 可以支持復雜的數據結構,而且帶有強大的數據查詢功能,因此非常受 到歡迎,很多項目都考慮用 MongoDB 來替代 MySQL 等傳統數據庫來實現不是特別復雜的 Web 應用。由于數據量實在太大,所以遷移到了 MongoDB 上面,數據查詢的速度得到了非 常顯著的提升。

MongoDB 基本操作詳解

下面將介紹一些查詢語法

1、 條件操作符

<, <=, >, >= 這個操作符就不用多解釋了 ,最常用也是最簡單的

db.collection.find({ "field" : { $gt: value } } ); // 大于: field > value
db.collection.find({ "field" : { $lt: value } } ); // 小于: field < value
db.collection.find({ "field" : { $gte: value } } ); // 大于等于: field >= value
db.collection.find({ "field" : { $lte: value } } ); // 小于等于: field <= value

如果要同時滿足多個條件,可以這樣做

db.collection.find({ "field" : { $gt: value1, $lt: value2 } } ); // value1 < field < value

2、$all 匹配所有

這個操作符跟 SQL 語法的 in 類似,但不同的是, in 只需滿足( )內的某一個值即可, 而$all 必須滿足[ ]內的所有值,例如:

db.users.find({age : {$all : [6, 8]}});

可以查詢出 {name: ‘David’, age: 26, age: [ 6, 8, 9 ] }
但查詢不出 {name: ‘David’, age: 26, age: [ 6, 7, 9 ] }

3、$exists 判斷字段是否存在

查詢所有存在 age 字段的記錄

db.users.find({age: {$exists: true}});

查詢所有不存在 name 字段的記錄

db.users.find({name: {$exists: false}});

舉例如下:

C1 表的數據如下:

>db.c1.find();
{ "_id" : ObjectId("4fb4a773afa87dc1bed9432d"), "age" : 20, "length" : 30 }
{ "_id" : ObjectId("4fb4a7e1afa87dc1bed9432e"), "age_1" : 20, "length_1" : 30 }

查詢存在字段 age 的數據

> db.c1.find({age:{$exists:true}});
{ "_id" : ObjectId("4fb4a773afa87dc1bed9432d"), "age" : 20, "length" : 30 }

可以看出只顯示出了有 age 字段的數據, age_1 的數據并沒有顯示出來

4、Null 值處理

Null 值的處理稍微有一點奇怪,具體看下面的樣例數據:

> db.c2.find()
{ "_id" : ObjectId("4fc34bb81d8a39f01cc17ef4"), "name" : "Lily", "age" : null }
{ "_id" : ObjectId("4fc34be01d8a39f01cc17ef5"), "name" : "Jacky", "age" : 23 }
{ "_id" : ObjectId("4fc34c1e1d8a39f01cc17ef6"), "name" : "Tom", "addr" : 23 }

其中 ”Lily”的 age 字段為空, Tom 沒有 age 字段,我們想找到 age 為空的行,具體如下:

> db.c2.find({age:null})
{ "_id" : ObjectId("4fc34bb81d8a39f01cc17ef4"), "name" : "Lily", "age" : null }
{ "_id" : ObjectId("4fc34c1e1d8a39f01cc17ef6"), "name" : "Tom", "addr" : 23 }

奇怪的是我們以為只能找到”Lily”,但”Tom”也被找出來了 ,所以”null”不僅能找到它自身 ,連不存在 age 字段的記錄也找出來了。那么怎么樣才能只找到”Lily”呢?我們用 exists 來限制一下即可:

> db.c2.find({age:{"$in":[null], "$exists":true}})
{ "_id" : ObjectId("4fc34bb81d8a39f01cc17ef4"), "name" : "Lily", "age" : null }

這樣如我們期望一樣,只有”Lily”被找出來了。

5、$mod 取模運算

查詢 age 取模 10 等于 0 的數據

db.student.find( { age: { $mod : [ 10 , 1 ] } } )

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

查詢 age 取模 6 等于 1 的數據

> db.c1.find({age: {$mod : [ 6 , 1 ] } })
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }

可以看出只顯示出了 age 取模 6 等于 1 的數據,其它不符合規則的數據并沒有顯示出來

6、$ne 不等于

查詢 x 的值不等于 3 的數據

db.things.find( { x : { $ne : 3 } } );

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

查詢 age 的值不等于 7 的數據

> db.c1.find( { age : { $ne : 7 } } );
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

可以看出只顯示出了 age 等于 7 的數據,其它不符合規則的數據并沒有顯示出來

7、$in 包含

與 sql 標準語法的用途是一樣的,即要查詢的是一系列枚舉值的范圍內查詢 x 的值在 2,4,6 范圍內的數據db.things.find({x:{$in: [2,4,6]}});

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

查詢 age 的值在 7,8 范圍內的數據

> db.c1.find({age:{$in: [7,8]}}); { "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }

可以看出只顯示出了 age 等于 7 或 8 的數據,其它不符合規則的數據并沒有顯示出來

8、$nin 不包含

與 sql 標準語法的用途是一樣的,即要查詢的數據在一系列枚舉值的范圍外查詢 x 的值在 2,4,6 范圍外的數據

db.things.find({x:{$nin: [2,4,6]}});

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

查詢 age 的值在 7,8 范圍外的數據

> db.c1.find({age:{$nin: [7,8]}});
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

可以看出只顯示出了 age 不等于 7 或 8 的數據,其它不符合規則的數據并沒有顯示出來

9、$size 數組元素個數

對于{name: ‘David’, age: 26, favorite_number: [ 6, 7, 9 ] }記錄匹配 db.users.find({favorite_number: {$size: 3}});不匹配 db.users.find({favorite_number: {$size: 2}});

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb4af85afa87dc1bed94330"), "age" : 7, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af89afa87dc1bed94331"), "age" : 8, "length_1" : 30 }
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

查詢 age 的值在 7,8 范圍外的數據

> db.c1.find({age:{$nin: [7,8]}});
{ "_id" : ObjectId("4fb4af8cafa87dc1bed94332"), "age" : 6, "length_1" : 30 }

可以看出只顯示出了 age 不等于 7 或 8 的數據,其它不符合規則的數據并沒有顯示出來

10、正則表達式匹配

查詢不匹配 name=B*帶頭的記錄db.users.find({name: {$not: /^B.*/}});

舉例如下:

C1 表的數據如下:

> db.c1.find();
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

查詢 name 不以 T 開頭的數據

> db.c1.find({name: {$not: /^T.*/}});
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

可以看出只顯示出了 name=Tony 的數據,其它不符合規則的數據并沒有顯示出來

11、Javascript 查詢和$where 查詢

查詢 a 大于 3 的數據,下面的查詢方法殊途同歸

 db.c1.find( { a : { $gt: 3 } } );
 db.c1.find( { $where: "this.a > 3" } );
 db.c1.find("this.a > 3");
 f = function() { return this.a > 3; } db.c1.find(f);

12、count 查詢記錄條數

count 查詢記錄條數db.users.find().count();

以下返回的不是 5,而是 user 表中所有的記錄數量

db.users.find().skip(10).limit(5).count();

如果要返回限制之后的記錄數量,要使用 count(true)或者 count(非 0)

db.users.find().skip(10).limit(5).count(true);

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

查詢 c1 表的數據量

> db.c1.count()
2

可以看出表中共有 2 條數據

13、skip 限制返回記錄的起點

從第 3 條記錄開始,返回 5 條記錄(limit 3, 5)

db.users.find().skip(3).limit(5);

舉例如下:

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

查詢 c1 表的第 2 條數據

> db.c1.find().skip(1).limit(1)
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

可以看出表中第 2 條數據被顯示了出來

14、sort 排序

以年齡升序 asc

db.users.find().sort({age: 1});

以年齡降序 desc

db.users.find().sort({age: -1});

C1 表的數據如下:

> db.c1.find()
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

查詢 c1 表按 age 升序排列

> db.c1.find().sort({age: 1});
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }

第 1 條是 age=10 的,而后升序排列結果集

查詢 c1 表按 age 降序排列

> db.c1.find().sort({age: -1});
{ "_id" : ObjectId("4fb5faaf6d0f9d8ea3fc91a8"), "name" : "Tony", "age" : 20 }
{ "_id" : ObjectId("4fb5fab96d0f9d8ea3fc91a9"), "name" : "Joe", "age" : 10 }

第 1 條是 age=20 的,而后降序排列結果集

15、游標

象大多數數據庫產品一樣, MongoDB 也是用游標來循環處理每一條結果數據,具體語法如下:

> for( var c = db.t3.find(); c.hasNext(); ) {
... printjson( c.next());
... }
{ "_id" : ObjectId("4fb8e4838b2cb86417c9423a"), "age" : 1 }
{ "_id" : ObjectId("4fb8e4878b2cb86417c9423b"), "age" : 2 }
{ "_id" : ObjectId("4fb8e4898b2cb86417c9423c"), "age" : 3 }
{ "_id" : ObjectId("4fb8e48c8b2cb86417c9423d"), "age" : 4 }
{ "_id" : ObjectId("4fb8e48e8b2cb86417c9423e"), "age" : 5 }

MongoDB 還有另一種方式來處理游標

> db.t3.find().forEach( function(u) { printjson(u); } );
{ "_id" : ObjectId("4fb8e4838b2cb86417c9423a"), "age" : 1 }
{ "_id" : ObjectId("4fb8e4878b2cb86417c9423b"), "age" : 2 }
{ "_id" : ObjectId("4fb8e4898b2cb86417c9423c"), "age" : 3 }
{ "_id" : ObjectId("4fb8e48c8b2cb86417c9423d"), "age" : 4 }
{ "_id" : ObjectId("4fb8e48e8b2cb86417c9423e"), "age" : 5 }
>

16、存儲過程

MongoDB 為很多問題提供了一系列的解決方案,針對于其它數據庫的特性,它仍然毫不示弱,表現的非比尋常。

MongoDB 同樣支持存儲過程。關于存儲過程你需要知道的第一件事就是它是用 javascript 來寫的。也許這會讓你很奇怪,為什么它用 javascript 來寫,但實際上它會讓你非常滿意,MongoDB 存儲過程是存儲在 db.system.js 表中的,我們想象一個簡單的 sql 自定義函數如下:

function addNumbers( x , y ) {
return x + y;
}

下面我們將這個 sql 自定義函數轉換為 MongoDB 的存儲過程:

> db.system.js.save({_id:”addNumbers”, value:function(x, y){ return x + y; }});

存儲過程可以被查看,修改和刪除,所以我們用 find 來查看一下是否這個存儲過程已經被創建上了 。

> db.system.js.find() 
{ "_id" : "addNumbers", "value" : function cf__1__f_(x, y) {
return x + y;
} }
>

這樣看起來還不錯,下面我看來實際調用一下這個存儲過程:

> db.eval('addNumbers(3, 4.2)');
7.2
>

這樣的操作方法簡直太簡單了 ,也許這就是 MongoDB 的魅力所在。
db.eval()是一個比較奇怪的東西,我們可以將存儲過程的邏輯直接在里面并同時調用,而無需事先聲明存儲過程的邏輯。

>db.eval( function() { return 3+3; } ); 6 >

從上面可以看出, MongoDB 的存儲過程可以方便的完成算術運算,但其它數據庫產品在存儲過程中可以處理數據庫內部的一些事情,例如取出某張表的數據量等等操作,這些MongoDB 能做到嗎?答案是肯定的, MongoDB 可以輕而易舉的做到,看下面的實例吧:

> db.system.js.save({_id:"get_count", value:function(){ return db.c1.count(); }});
> db.eval('get_count()')
2
 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!