版本一:! u/ b0 E; R# ~! e3 F# S0 `' ?2 G3 J) R
0 e9 \1 R0 o7 ~/ | {8 q2 K
1 ) . 大于,小于,大于或等于,小于或等于
0 C7 J$ C3 S g/ [, Z' x5 _: w ~# W/ o8 O0 t
$gt:大于% \9 j5 S5 y0 X# ^& S! ` r% U/ j
$lt:小于
3 F+ o: V* Q% G# f$gte:大于或等于( N4 y: x- ?) K
$lte:小于或等于% ]- r6 w. r( |. `* _- A
( ^0 J8 M/ O" y$ [例子: - db.collection.find({ "field" : { $gt: value } } ); // greater than : field > value
! j5 V4 y' y5 Q( q - db.collection.find({ "field" : { $lt: value } } ); // less than : field < value7 S. w x: O# f8 a! L
- db.collection.find({ "field" : { $gte: value } } ); // greater than or equal to : field >= value9 t( h7 |$ {) A6 e
- db.collection.find({ "field" : { $lte: value } } ); // less than or equal to : field <= value
复制代码
0 z' i3 v6 y: y2 j如查询j大于3,小于4: - db.things.find({j : {$lt: 3}});
% y! S2 Y4 x y* S! K& ?1 T - db.things.find({j : {$gte: 4}});
复制代码
4 G4 y/ b; a# y2 g0 e p, u也可以合并在一条语句内: - db.collection.find({ "field" : { $gt: value1, $lt: value2 } } ); // value1 < field < value
复制代码 . f! x* b0 ]6 d2 b" z/ }
4 c8 p$ h" S0 Y' T- e6 q2 u
: Y, g2 _ @7 @2 a" G0 n- M2 a2) 不等于 $ne 例子: - db.things.find( { x : { $ne : 3 } } );
复制代码 : ^% E1 l3 H/ W% k; j1 R( r
3) in 和 not in ($in $nin)( x5 `7 C8 r" {! r: R G
6 G4 t, a5 d/ p5 D0 G' `% _
语法: - db.collection.find( { "field" : { $in : array } } );
复制代码 & J, c2 ]' F5 Y1 S# T0 ` c
例子: - db.things.find({j:{$in: [2,4,6]}});
0 ?+ B& G: c- C' ?& ^$ m - db.things.find({j:{$nin: [2,4,6]}});
复制代码 1 l8 p, f; O. i% O
6 f! V% J* p( S8 R2 k7 ?& b4) 取模运算$mod
+ M' b% w1 P/ y( Q/ U
6 u& g" N. k& S如下面的运算:
- db.things.find( "this.a % 10 == 1")
复制代码 : q$ N$ Z) ^$ j
可用$mod代替: - db.things.find( { a : { $mod : [ 10 , 1 ] } } )
复制代码 ! r$ V' |! G/ F7 j
% M# E( b3 T* n3 N: e" `
5) $all( G6 M. U$ v" I- v" |7 y
; r* k5 K- N, M; d; K, d/ e
$all和$in类似,但是他需要匹配条件内所有的值:
! R1 V3 J# l* Z% P: Q# n7 \1 ^8 _) @! A C% Q" F
如有一个对象:
; \ H" f" g2 I : Q# w( u& {$ v5 N
下面这个条件是可以匹配的: - db.things.find( { a: { $all: [ 2, 3 ] } } );
复制代码
$ {6 M& y. m4 V但是下面这个条件就不行了: - db.things.find( { a: { $all: [ 2, 3, 4 ] } } );
复制代码 1 `; {; A8 C: v2 G* V6 W
2 _: g3 L# v. U; N
6) $size- b* w3 O* S: u! m8 Z3 E
/ [; \' J) E6 e, H
$size是匹配数组内的元素数量的,如有一个对象:{a:["foo"]},他只有一个元素:
4 T/ K3 C% Y7 Y9 a5 s* G X1 H
- P# D- y: N' ?下面的语句就可以匹配: - db.things.find( { a : { $size: 1 } } );
复制代码
C; y5 n) E3 K( B2 F3 q官网上说不能用来匹配一个范围内的元素,如果想找$size<5之类的,他们建议创建一个字段来保存元素的数量。 You cannot use $size to find a range of sizes (for example: arrays with more than 1 element). If you need to query for a range, create an extra size field that you increment when you add elements. * E4 j! K5 H7 G5 Z4 S1 U3 ^' p$ q
7)$exists $exists用来判断一个元素是否存在: 如: - db.things.find( { a : { $exists : true } } ); // 如果存在元素a,就返回9 B H( U3 Y2 {, E
- db.things.find( { a : { $exists : false } } ); // 如果不存在元素a,就返回
复制代码 % S9 N C! {7 U8 s9 e/ g
8) $type $type 基于 bson type来匹配一个元素的类型,像是按照类型ID来匹配,不过我没找到bson类型和id对照表。 - db.things.find( { a : { $type : 2 } } ); // matches if a is a string
' Z+ t, C" i1 t [8 g3 q5 O! V( z - db.things.find( { a : { $type : 16 } } ); // matches if a is an int
复制代码
/ {! Z: [4 p+ }/ _) k0 h- j/ z9)正则表达式2 c3 v9 e8 R# o/ `% _7 w
5 w" e9 b- I7 u/ ?( ]
mongo支持正则表达式,如: - db.customers.find( { name : /acme.*corp/i } ); // 后面的i的意思是区分大小写
复制代码
7 I/ N3 F- }, k- s10) 查询数据内的值
3 N8 d7 I; p7 a2 S9 ^3 C- V% M0 o) H7 s: W. C
下面的查询是查询colors内red的记录,如果colors元素是一个数据,数据库将遍历这个数组的元素来查询。 - db.things.find( { colors : "red" } );
复制代码 ( x& l5 ]! T: }6 U6 P. V
11) $elemMatch' \+ v2 O; B- _+ d% Q# L" h- u
: @$ i! F& v: y4 o, }1 [如果对象有一个元素是数组,那么$elemMatch可以匹配内数组内的元素: - > t.find( { x : { $elemMatch : { a : 1, b : { $gt : 1 } } } } )
" ^ N! ~( W# z9 X - { "_id" : ObjectId("4b5783300334000000000aa9"), 7 f4 _0 K) d+ U# y1 m. Y8 b
- "x" : [ { "a" : 1, "b" : 3 }, 7, { "b" : 99 }, { "a" : 11 } ]) F. i* W! ]* x9 W- j' Q! W6 C$ B" I
- }
复制代码
5 e0 y& Y$ p( p8 p5 n, z. r$elemMatch : { a : 1, b : { $gt : 1 } } 所有的条件都要匹配上才行。注意,上面的语句和下面是不一样的。 > t.find( { "x.a" : 1, "x.b" : { $gt : 1 } } )
/ d0 ^1 k( U- V. a9 d$elemMatch是匹配{ "a" : 1, "b" : 3 },而后面一句是匹配{ "b" : 99 }, { "a" : 11 }
w0 E: J# P3 J9 r$ w2 k9 x2 ~; s" @( p
12) 查询嵌入对象的值 - db.postings.find( { "author.name" : "joe" } );
复制代码
: d$ O3 i; `( ?# M/ m! S5 d$ S举个例子: - > db.blog.save({ title : "My First Post", author: {name : "Jane", id : 1}})
复制代码
" {1 s/ C# K& W( p$ h如果我们要查询 authors name 是Jane的, 我们可以这样: - > db.blog.findOne({"author.name" : "Jane"})
复制代码
% y; w+ a" s! Z9 y* L如果不用点,那就需要用下面这句才能匹配: - db.blog.findOne({"author" : {"name" : "Jane", "id" : 1}})
复制代码
: P9 \4 ]! n& B2 _" Z下面这句: - db.blog.findOne({"author" : {"name" : "Jane"}})
复制代码
! l6 S9 i ^) U/ q5 J是不能匹配的,因为mongodb对于子对象,他是精确匹配。
* \. A: ?; f2 \) D13) 元操作符 $not 取反 如: - db.customers.find( { name : { $not : /acme.*corp/i } } );# j3 R! _6 Q! {" [4 Q% K( P& X
- db.things.find( { a : { $not : { $mod : [ 10 , 1 ] } } } );
复制代码
5 X! ^2 t8 m7 M5 `" ashell 环境下的操作: 1. 超级用户相关: 1. #进入数据库admin
5 b/ e; v: ?, u* y, u, Y
- f( ]/ F& r2 ?+ c, ? 2. #增加或修改用户密码 3. #查看用户列表 4. #用户认证 5. #删除用户 ; ?3 k, F0 G% I. J9 C9 D2 P
6. #查看所有用户 7. #查看所有数据库
" M3 P8 |* Q: i$ a& i* e3 c 8. #查看所有的collection 9. #查看各collection的状态 - db.printCollectionStats()
复制代码
) s$ W+ } E* L2 Y, h 10. #查看主从复制状态 - db.printReplicationInfo()
复制代码
+ T% X2 }2 ]0 N1 g 11. #修复数据库 12. #设置记录profiling,0=off 1=slow 2=all 13. #查看profiling 14. #拷贝数据库 - db.copyDatabase('mail_addr','mail_addr_tmp')
复制代码 4 j$ x- ]2 D* @+ W
15. #删除collection 16. #删除当前的数据库 2. 增删改 1. #存储嵌套的对象 - db.foo.save({'name':'ysz','address':{'city':'beijing','post':100096},'phone':[138,139]})
复制代码
: h' p1 G2 b) q# r9 X) T+ f 2. #存储数组对象 - db.user_addr.save({'Uid':'yushunzhi@sohu.com','Al':['test-1@sohu.com','test-2@sohu.com']})
复制代码
0 M' Z1 u5 s8 { 3. #根据query条件修改,如果不存在则插入,允许修改多条记录 - db.foo.update({'yy':5},{'$set':{'xx':2}},upsert=true,multi=true)
复制代码 - A, ?, K1 A8 F: `/ H
4. #删除yy=5的记录 5. #删除所有的记录 3. 索引 1. #增加索引:1(ascending),-1(descending) 2. db.foo.ensureIndex({firstname: 1, lastname: 1}, {unique: true}); 3. #索引子对象 4. db.user_addr.ensureIndex({'Al.Em': 1}) 5. #查看索引信息 6. db.foo.getIndexes() 7. db.foo.getIndexKeys() 8. #根据索引名删除索引 9. db.user_addr.dropIndex('Al.Em_1') 4. 查询 1. #查找所有 2. db.foo.find() 3. #查找一条记录 4. db.foo.findOne() 5. #根据条件检索10条记录 6. db.foo.find({'msg':'Hello 1'}).limit(10) 7. #sort排序 8. db.deliver_status.find({'From':'ixigua@sina.com'}).sort({'Dt',-1}) 9. db.deliver_status.find().sort({'Ct':-1}).limit(1) 10. #count操作 11. db.user_addr.count() 12. #distinct操作,查询指定列,去重复 13. db.foo.distinct('msg') 14. #”>=”操作 15. db.foo.find({"timestamp": {"$gte" : 2}}) 16. #子对象的查找 17. db.foo.find({'address.city':'beijing'}) 5. 管理 1. #查看collection数据的大小 2. db.deliver_status.dataSize() 3. #查看colleciont状态 4. db.deliver_status.stats() 5. #查询所有索引的大小 6. db.deliver_status.totalIndexSize() , V' `. B$ z9 A& c( p1 w1 t
6. 高级查询 条件操作符 % u# M7 |2 T3 P: x, u. M, h: }$ {, c# v* w! H
- $gt : >
6 B% g! z8 D- E7 p5 k6 ]7 K - $lt : <
/ @& n* t! o$ \" S+ t - $gte: >= ! \/ M3 H1 L# |: v7 |7 V& T
- $lte: <= - \; B+ u T0 r. s3 }+ {4 d
- $ne : !=、<> 0 r5 P( [7 ]4 b, S9 z' W/ J7 f
- $in : in
* U) b' ^2 V# X C: y( a5 G# T, _ - $nin: not in
' T' ^% b: h" u0 y - $all: all
# p7 G' A# _ I" P! i4 a - $not: 反匹配(1.3.3及以上版本)
复制代码 0 ?! u# S5 g; b% e2 j
3 p2 _2 X9 S# K4 s7 N
查询 name <> "bruce" and age >= 18 的数据
# X$ |- w; `/ ]- Y% v3 \- db.users.find({name: {$ne: "bruce"}, age: {$gte: 18}});
复制代码
- P B z' Y: V, W" I; B4 {: K$ q
8 T0 x1 C" u3 u) d. T, _查询 creation_date > '2010-01-01' and creation_date <= '2010-12-31' 的数据
( r5 h9 F- y8 G2 q$ h# z- db.users.find({creation_date:{$gt:new Date(2010,0,1), $lte:new Date(2010,11,31)});
复制代码 * ^' B! q8 @, w
! }/ o- q: H; ~, L4 I
查询 age in (20,22,24,26) 的数据
& K- E9 Z h. }' J8 K- db.users.find({age: {$in: [20,22,24,26]}});
复制代码 + p" c9 |3 Z: ^: X
$ h( B/ q7 c2 B: h3 b# F% M
查询 age取模10等于0 的数据 1 K" U# N( l0 _5 o* c
- db.users.find('this.age % 10 == 0');
复制代码
_& e/ H1 J; C0 Y; Q: U( U0 u2 N6 c3 Q或者 1 X% E _% k3 f+ {% {* h) D3 s* v$ r
- db.users.find({age : {$mod : [10, 0]}});
复制代码 + G3 ?) _1 s% \3 @
, J+ r: n. T* U
匹配所有 ( @8 h+ s, w% {" g1 a* x
- db.users.find({favorite_number : {$all : [6, 8]}});
复制代码
a8 b7 s; A( R. t6 z0 y- E7 T可以查询出{name: 'David', age: 26, favorite_number: [ 6, 8, 9 ] } % Z7 a( ^1 w1 J& V7 G
可以不查询出{name: 'David', age: 26, favorite_number: [ 6, 7, 9 ] } 3 H! T+ t" G( \1 k- ?* R1 R! M7 W* g
- [- x; P1 x/ ~$ ]查询不匹配name=B*带头的记录 4 l9 U7 i2 y0 c! k0 K$ ~1 u% `% m
- db.users.find({name: {$not: /^B.*/}});
复制代码 + f; z) Y4 b. {4 @9 U- A
查询 age取模10不等于0 的数据 0 {5 m0 S! H: N! P* h
- db.users.find({age : {$not: {$mod : [10, 0]}}});
复制代码 ; x* U+ ?1 x; g, W- c
: Y3 H1 k# w1 m
#返回部分字段
# M3 N% i6 m/ D) P" s' [选择返回age和_id字段(_id字段总是会被返回)
6 Q# C" K2 ^" h. G8 v% ~- db.users.find({}, {age:1}); # k: H4 C6 `) L" j" e
- db.users.find({}, {age:3}); ! J$ Z9 H0 g- p- ]9 q9 |5 e; D+ ]
- db.users.find({}, {age:true});
9 g' H! [8 Q. K7 j& D3 t$ k1 j - db.users.find({ name : "bruce" }, {age:1});
复制代码
$ p: `( M3 t. ? H4 f/ }8 [- K; p0为false, 非0为true ( S/ }* z9 n* j6 u; I
: x& a' j/ W9 E3 l选择返回age、address和_id字段 3 `! _* {! C# V5 @$ _
- db.users.find({ name : "bruce" }, {age:1, address:1});
复制代码
. K. _7 k) X2 t4 C
' x; N) b! C! L& \, P$ ^排除返回age、address和_id字段 . O4 T, l6 D5 S `
- db.users.find({}, {age:0, address:false}); 7 ?9 `$ A) A8 v
- db.users.find({ name : "bruce" }, {age:0, address:false});
复制代码 . @6 A, w5 U9 N8 f* I! Y; _/ x; O
6 _) h4 {7 |" A5 b) d5 z$ Y4 x. _
数组元素个数判断
& c* P6 h2 u) \$ j- R3 o" Q- a- w0 N对于{name: 'David', age: 26, favorite_number: [ 6, 7, 9 ] }记录
1 v$ D n1 B# X. m+ ^匹配db.users.find({favorite_number: {$size: 3}});
* V6 v1 u2 i. U% L; h不匹配db.users.find({favorite_number: {$size: 2}}); * c. C3 h% ^, J W) w
- b. i3 _5 i- \
$exists判断字段是否存在
6 |/ `% q! P& K' j# W8 z6 _6 b# x查询所有存在name字段的记录 6 ?' ]/ ]: m* q4 ]! B9 u3 x- ^
- db.users.find({name: {$exists: true}});
复制代码
! j5 _4 d, t8 a1 N查询所有不存在phone字段的记录 % e8 A! j B8 ?
- db.users.find({phone: {$exists: false}});
复制代码
' c( C( w* X/ e6 G$ r9 E+ b" N4 j& A6 ?9 C3 ~0 I; r. T, Z
$type判断字段类型 % S1 B- y2 F. t% X6 E6 r+ s/ x
查询所有name字段是字符类型的
3 X/ g# l; z- I" K d f4 w- db.users.find({name: {$type: 2}}); % D# F0 s/ \, G# U6 z" _8 O# c5 l/ |
复制代码
# o2 h- y, n5 K+ ?2 n! o: F. k" f查询所有age字段是整型的 8 u3 m* }( i( q0 T; d% H
- db.users.find({age: {$type: 16}});
, F/ S& X5 v, t; c& J
复制代码
Y' Z. f$ L9 h对于字符字段,可以使用正则表达式 - f- O' R F$ b' f
查询以字母b或者B带头的所有记录 , l L6 k; i. k' ^# |
- db.users.find({name: /^b.*/i}); # Q0 `5 b. L" c! Q: J$ z
复制代码
6 x. ^6 h8 n2 t$elemMatch(1.3.1及以上版本) 6 q+ T# v% b8 d
为数组的字段中匹配其中某个元素 - V! D8 _) O! Y6 J
" m0 h. }) ?9 I ^6 ^! o
Javascript查询和$where查询 % z+ {; J5 _3 d9 W" a s. W
查询 age > 18 的记录,以下查询都一样 8 D: ` x3 R0 C
- db.users.find({age: {$gt: 18}});
/ O+ g# |- O7 |* c3 h - db.users.find({$where: "this.age > 18"});
! ~0 m3 ], u( {% Q, P - db.users.find("this.age > 18"); 2 c4 T$ _+ x9 R$ L, w
- f = function() {return this.age > 18} db.users.find(f);
复制代码 * `* O I4 a; T: d8 b0 F
& o5 p6 r; x' b排序sort()
% N1 A W) u8 X5 J: O3 D' D* |/ x以年龄升序asc
^- R7 p1 z% f/ D" c' k2 f& m- db.users.find().sort({age: 1});
; B0 @2 t6 b- _% O
复制代码
y/ B' }! Y8 r! Y3 [以年龄降序desc
3 Z4 g8 `' [$ a8 }$ [* v c! J5 `" y- db.users.find().sort({age: -1}); * w1 j9 L, m9 B( L$ K
复制代码 + w6 p# H L6 H3 j& v- E- V
限制返回记录数量limit() Y; N2 k7 i( W: k( B
返回5条记录
7 A3 u# f- |( I+ q- db.users.find().limit(5);
1 z& N% U' y% P2 I: c
复制代码
6 E( @ G# T2 r* X4 K1 M4 U L# ^+ f返回3条记录并打印信息
+ B. Y0 \# @0 t. w& {6 n" l& e* ~: G- db.users.find().limit(3).forEach(function(user) {print('my age is ' + user.age)}); % p) a$ X( V; A6 j2 q
复制代码 $ X/ e; N" \% l
结果 + t! R7 K1 C+ h# ]
- my age is 18 8 r# S1 |7 s0 y- f
- my age is 19
& ]& j6 ] H" A8 P6 @# T - my age is 20
复制代码 ! b$ r* _% [! v: X4 o
2 d- F$ X0 z6 r$ X限制返回记录的开始点skip()
9 V; I) _8 y! k. F! ?; l从第3条记录开始,返回5条记录(limit 3, 5) ) q1 j% g2 L- B( L& d
- db.users.find().skip(3).limit(5);
. ` \6 B# `" ]) O+ S% U8 @ H
复制代码 0 I7 ?* w& q% m' C
查询记录条数count()
, `2 r" u" h8 l0 M: [7 \! c/ Z" Udb.users.find().count(); 5 t7 C2 v5 M& y, D6 r
db.users.find({age:18}).count();
. ?7 ]4 W$ L0 a$ r" M以下返回的不是5,而是user表中所有的记录数量 1 n C) S+ E! ]
db.users.find().skip(10).limit(5).count(); . W* j& {( Z8 l: ]$ d, v& a
如果要返回限制之后的记录数量,要使用count(true)或者count(非0)
. B8 N& \, ?! |$ R; W% o6 W: g0 Y7 W- db.users.find().skip(10).limit(5).count(true);
复制代码 # A: L, w9 ]9 j* K* G& J. y
: q' @7 I1 P6 L" A7 W$ S& R分组group() J8 P3 Z, j! x: C0 n
假设test表只有以下一条数据
7 Q2 o3 Q* U5 x7 ~% a- { domain: "www.mongodb.org" $ [9 O9 L+ w; _1 b
- , invoked_at: {d:"2009-11-03", t:"17:14:05"} 2 B- f0 W# n# I0 b! i) a" t: t
- , response_time: 0.05 8 H8 J" b4 N. L3 L* g1 G
- , http_action: "GET /display/DOCS/Aggregation"
2 ~9 {2 g6 i& a F - }
复制代码
! F% S+ q) j! u3 `& O使用group统计test表11月份的数据count:count(*)、total_time:sum(response_time)、avg_time:total_time/count;
1 c; x9 F! q, Y5 @- db.test.group( 2 y* b* P" `2 J# T/ l2 n! Y
- { cond: {"invoked_at.d": {$gt: "2009-11", $lt: "2009-12"}} 8 |, o. V) e- _1 A, Y
- , key: {http_action: true}
9 e& @, T0 r, N; J& O7 x- m - , initial: {count: 0, total_time:0}
0 v9 ^# w' u9 h, u* K/ [: C - , reduce: function(doc, out){ out.count++; out.total_time+=doc.response_time } : d4 a! V. Z+ G w
- , finalize: function(out){ out.avg_time = out.total_time / out.count }
y V! T! ?- a$ @( U$ g - } );
}6 b% `% v1 ~5 [+ a2 n' ` - + S, T$ ?; a3 R8 T$ \) C- `( i7 }
- [
1 d" D% K- \" `: p, ]/ X - { G, |) j4 F I
- "http_action" : "GET /display/DOCS/Aggregation",
2 n: Z5 z; H" y F" [4 ^( B - "count" : 1,
% S4 `" w0 l1 i( O# O! y4 g - "total_time" : 0.05,
; h; n& k; E7 s. w' w7 y, R - "avg_time" : 0.05
! c/ V4 W! ?, ?+ {5 Y; G' `% v - }
# ~" V: E8 e# ]" D+ Y+ n% P* R" V' {* Y - ]
复制代码
# C, ^7 r1 E- ]! d8 w* }# E7 x4 M# v3 k) }2 |4 p7 g
3 K1 C: E! y' D1 m5 U0 C2 cMongoDB 高级聚合查询MongoDB版本为:2.0.8 系统为:64位Ubuntu 12.04 先给他家看一下我的表结构[Oh sorry, Mongo叫集合] 如你所见,我尽量的模拟现实生活中的场景。这是一个人的实体,他有基本的manId, manName, 有朋友[myFriends],有喜欢的水果[fruits],而且每种水果都有喜欢的权重。 很不好的是你还看见了有个“_class”字段? 因为我是Java开发者, 我还喜欢用Spring,因此我选用了Spring Data Mongo的类库[也算是框架吧,但是我不这么觉得]。 现在有很多人Spring见的腻了也开始烦了。是的,Spring野心很大,他几乎想要垄断Java方面的任何事情。没办法我从使用Spring后就离不开他,以至于其他框架基本上都不用学。我学了Spring的很多,诸如:Spring Security/Spring Integration/Spring Batch等。。。不发明轮子的他已经提供了编程里的很多场景,我利用那些场景解决了工作中的很多问题,也使我的工作变得很高效。从而我又时间学到它更多。Spring Data Mongo封装了mongodb java driver,提供了和SpringJDBC/Template一致编程风格的MongoTemplate。 不说废话了,我们直接来MongoDB吧。 - Max 和Min
3 L; c* B$ _9 \# T0 I1 W- V
我和同事在测试Mongo时,索引还写了不到一半,他想查询某个字段的最大值,结果找了半天文档也没找到关于max的函数。我也很纳闷这是常规函数啊怎么不提供? 后来经过翻阅资料确定Mongo确实不提供直接的max和min函数。但是可以通过间接的方式[sort 和 limit]实现这个。 要查询最大值我们只需要把结果集按照降序排列,取第一个值就是了。 如我的例子,我想取得集合中年龄最大的人。 - db.person.find({}).sort({"age" : -1}).limit(1)
复制代码 6 B( p j+ ]4 Y! V) _1 @/ @) h
0 r* k" C. `# c/ D* V
2 e: D0 D9 }# i* A$ w相反如果想要年龄最小的人,只需要把sort中改为{“age”:1}就可以了。 当然我们使用了sort,对于小数量的文档是没问题的。当对于大量数据需要给age建立索引,否则这个操作很耗时。 - distinct v4 B3 w4 H; H! h \2 j1 N
MongoDB的destinct命令是获取特定字段中不同值列表的最简单工具。该命令适用于普通字段,数组字段[myFriends]和数组内嵌文档[fruits]. 如上面的图片,我认为fruits和myFriends字段是不同的。网上很多资料和例子都没说到这个情景,因为我们也业务是fruits这样的模型,我测试了。对于fruits.fruitId他也是可行的。 如上面的表结构,我想统计所有的喜欢的水果。 - db.person.distinct("fruits.fruitId") // 查找对象里引入对象的值,直接加.
复制代码
$ }) u& H2 Z3 q; P! v8 \- H
* Q5 H3 O0 b. Z6 n2 j! X 他成功执行了。输出如: - [ "aaa", "bbb", "ccc", "www", "xxx", "yyy", "zzz", "rrr" ]
复制代码 . L) I# ?% B$ {; u
' \0 G$ ]8 a4 g" E% \, j* b
我想统计集合中共有多少个人[按名字吧] - db.person.distinct("manName")
复制代码
2 S5 X9 A2 b7 [; H; \6 P& q
# I, w8 u, `0 t3 |2 E2 u( L 我想统计指定个数的人的共同关注的朋友。 - db.person.distinct("myFriends", {"manName" : {"$in" : ["ZhenQin", "YangYan"]}})
复制代码 4 Y" Y: A/ e2 r+ k" @& q
) C; p; X" ?8 N# R& P
输出如: -
3 u, h9 ]: w- V% A8 n- k5 k* `& A2 A - [ "234567", "345678", "456789", "987654", "ni", "wo" ]
复制代码 7 k9 s# _( ~# T* ^- ], [
& V5 T* ~7 |# p( W. [1 ^
G3 G7 b C! ~: ?
那么我使用Java呢? 我只是在演示Mongo的命令,用Spring Data Mongo是怎么操作的? Spring Schema: - <beans xmlns="http://www.springframework.org/schema/beans"
8 z+ m4 q- ?' V$ h5 u) A - xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
8 S/ p1 ^7 {7 F- V+ N - xmlns:context="http://www.springframework.org/schema/context"8 m6 ^7 M' x/ V& o, W
- xmlns:mongo="http://www.springframework.org/schema/data/mongo"8 I; ~( ]8 ?0 C0 X( {8 r
- xsi:schemaLocation="http://www.springframework.org/schema/beans( i4 Z4 o% ^# L
- http://www.springframework.org/schema/beans/spring-beans-3.0.xsd! U0 M5 Z% o9 ~2 ~% E
- http://www.springframework.org/schema/context! h' D/ Q9 o" w% h# C; f% T# l3 x
- http://www.springframework.org/schema/context/spring-context-3.1.xsd
1 x6 c5 j/ ~, A! s - http://www.springframework.org/schema/data/mongo* d& S9 h8 l# Q
- http://www.springframework.org/schema/data/mongo/spring-mongo-1.0.xsd">3 f% ]& y6 k7 l5 x+ _% ^2 v% r2 \* L
-
& M3 P3 y, f8 y. ^2 b5 Q7 H - <context:property-placeholder location="classpath:mongo.properties" />
1 s. d' X; s9 ]8 V2 w -
- u) C. o' ~" {0 E/ j - <!-- Default bean name is 'mongo' -->
* n* k6 P2 `( ?8 s& N$ q9 u. ?! |2 b - <mongo:mongo id="mongo" host="${mongo.host}" port="${mongo.port}" />
6 e5 a p( F) E8 P: k - 9 d& B' R+ f8 {$ w
- <mongo:db-factory id="mongoDbFactory"3 l; @5 t B7 I
- mongo-ref="mongo"
: l3 p) f( P/ W6 S3 H9 x7 i - dbname="mongotest" />3 C2 t6 X0 t, z* {4 [
-
b$ g% \0 w6 x9 }6 u - <bean id="mongoTemplate" class="org.springframework.data.mongodb.core.MongoTemplate"># d5 _- [; `) n& r; w7 c; i
- <constructor-arg name="mongoDbFactory" ref="mongoDbFactory"/>
) F* w7 }$ Q, v& P+ I1 A - </bean>
+ Q* V- W# `0 C- R, ^8 _. ? - </beans>
复制代码
/ R# U$ ?1 u8 K# M
; a" S% O) H1 y g) x" i6 e q2 T max和min的测试: - @Test2 f! ?% r# u2 g- g' U
- public void testMaxAndMinAge() throws Exception {; E- c$ z3 |$ a4 L3 r: t
- Query q = new BasicQuery("{}").with(new Sort(new Sort.Order(Sort.Direction.ASC, "age"))).limit(1);
* D8 o6 n$ R% }/ K) @ - Person result = mongoTemplate.findOne(q, Person.class);
# W. E* d/ a- E: v - log.info(result);* v& b# I7 B: [3 H( ~0 Q% W S/ j
- ) v4 x$ s4 |! d
- q = new BasicQuery("{}").with(new Sort(new Sort.Order(Sort.Direction.DESC, "age"))).limit(1);$ Q+ L5 j6 p1 p5 X/ j# W
- result = mongoTemplate.findOne(q, Person.class);
1 y1 @, Z5 i" m5 c5 J( ^9 L - log.info(result);: C% @3 U% k6 m! H$ U
- }
复制代码 # N0 Q: r0 P! i9 N' Y J; o7 w! ~+ Q
! v% U5 Y/ Z: D) l" d. j1 v
distinct的测试: - @Test
$ t5 z2 K% b+ C - public void testDistinct() throws Exception {& w: i' T( m. A( G
- List result = mongoTemplate.getCollection("person").distinct("myFriends");
, v, p9 |" h* c8 l2 u% _; u - for (Object o : result) {9 K2 n1 _( U4 ]; N) U L
- log.info(o);
& |9 U9 X2 k7 C; j - }, @2 ]( g3 k, @" m* k* d$ I0 |
-
' r8 l4 R# b7 `& t* b5 r3 |2 y' w5 A. y - log.info("==================================================================");
, _$ o5 \5 c0 Y& x - Query query = Query.query(Criteria.where("manId").is("123456"));% J% Y/ E7 B7 s x/ l
- result = mongoTemplate.getCollection("person").distinct("myFriends", query.getQueryObject());) I# B- C1 g4 a
- for (Object o : result) {
, {6 r8 x# o- A" u& l9 R - log.info(o);
+ w0 |9 {1 t/ X - }0 [' H3 i. @: g& b* o5 H, W
- $ [; i& s% l! h5 S! H5 X) J7 ^
- log.info("==================================================================");7 _* f6 a J; i& k5 q3 W
- result = mongoTemplate.getCollection("person").distinct("fruits.fruitId");- B0 C( c8 l8 [
- for (Object o : result) {2 @5 ^& z% d, x G0 p9 ~
- log.info(o);7 t4 T5 {3 }+ L. b# V( ]; M2 m
- } D# E3 B J1 i! |. N( n
- }
复制代码 % Q; t. t" G; D; T8 j. a
7 q8 Z1 U$ F; l# {) u l
输出的结果为: - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] 234567: a, [; ]4 x1 o: u: F
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] 345678
- i2 A5 `# w! w8 e: K - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] 456789; f6 M9 B# M! a0 |
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] 9876541 T" m' `- n5 }3 s/ E; B" @
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] ni
$ e# L: `* y7 t4 w# s' T* h/ H3 ^ - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] wo- H" `! ~+ u. E9 @# `! }
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(72)] 1234565 f# u, k+ U! u* \# ~9 j
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(75)] ==================================================================4 ~) m0 V8 y9 Y- I, z! S
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(79)] 234567" \, B% C0 \8 Y
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(79)] 3456781 `0 P2 j/ L* O+ v4 N
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(79)] 456789
+ e# _" E% Y, D! Q7 S8 }; T - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(79)] 987654* G0 P0 z. Z, B i! D& e* e, r
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(82)] ==================================================================1 T0 C. ?* |2 ~$ q; Q5 h9 ?0 `
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] aaa
8 K$ c0 d" O' ]( t; r/ m - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] bbb+ g9 {5 [( W. k2 F: w1 b: p
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] ccc" L& V& o4 u, \1 z( t. S
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] www A0 K( l2 n) o- z- Y
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] xxx) L' @ ] P8 k2 E
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] yyy
4 l3 o \, u0 h5 n- P2 \6 L - 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] zzz: P0 Z; K. c% ~$ ^" m
- 12-22 14:13:45 [INFO] [t.MongoAdvaceQueryTest(85)] rrr& S" g G! }; D. A2 o0 R& J
- 12-22 14:13:45 [INFO] [support.GenericApplicationContext(1020)] Closing org.springframework.context.support.GenericApplicationContext@1e0a91ff: startup date [Sat Dec 22 14:13:44 CST 2012]; root of context hierarchy
复制代码
1 j R9 k5 {/ U" k: O# T8 g
1 a3 @1 y2 Y9 y$ y( F- p; z 这里我要特别说明一下, 当使用了Spring Data Mongo,如上面的findOne(query, Person.class)它就会把查询的结果集转换成Person类的对象。Spring Data Mongo的很多API中都这样,让传入了一个Bean的class对象。因为distinct的测试是输出list<String>的,我 使用的mongo-java-driver的api。他们都很简单,唯一的是Query这个Spring提供的对象,希望读者注意,他几乎封装了所有条件 查询,sort,limit等信息。
7 E( M9 Q) T# k" z# @( d& d& e6 j2 u; _7 C% Q
% m r$ |8 K" c* o! T% R0 M4 f
|