02-mongodb基本命令操作

02-mongodb基本命令操作

一、简介

mongodb 文档数据库,存储的是文档(Bson->json的二进制化).

特点 : 内部执行引擎为JS解释器, 把文档存储成bson结构,在查询时,转换为JS对象,并可以通过熟悉的js语法来操作.

mongo和传统型数据库相比,最大的不同:

传统型数据库

结构化数据, 定好了表结构后,每一行的内容,必是符合表结构的,就是说--列的个数,类型都一样.

mongo文档型数据库:

表下的每篇文档,都可以有自己独特的结构(json对象都可以有自己独特的属性和值)

举例:如果有电影, 影评, 影评的回复, 回复的打分

在传统型数据库中, 至少要4张表, 关联度非常复杂.

在文档数据库中,通过1篇文档,即可完成. 体现出文档型数据库的反范式化.

{
    movie:'天龙八部'
    comment:[
        {content:'王家卫的电影风格',reply:['支持','好']}
    ]
}

1.1 mongodb的安装

1: 下载mongodb www.mongodb.org 版本号偶数为稳定版,奇数为开发版

2: 安装参考上一篇我的博客 01-mongodb副本集架构配置_q287573145的博客-CSDN博客

3:bin目录下说明

1.2 mongo入门命令

show dbs  查看当前的数据库
use databaseName 选库
show tables/collections 查看当前库下的collection
如何创建库?
    Mongodb的库是隐式创建,你可以use 一个不存在的库,然后在该库下创建collection,即可创建库
创建collection
    db.createCollection('collectionName')  
collection允许隐式创建
    db.collectionName.insert(document)
删除collection
    db.collectionName.drop()
删除database
    db.dropDatabase();

1.2.1 增:insert

介绍: mongodb存储的是文档, 文档指的是json格式的对象
语法: db.collectionName.isnert(document);
​
例1:增加单篇文档
    db.collectionName.insert({title:'nice day'});
例2:增加单个文档,并指定_id
    db.collectionName.insert({_id:8,age:78,name:'lisi'});
例3:增加多个文档
    db.collectionName.insert(
        [
            {time:'friday',study:'mongodb'},
            {_id:9,gender:'male',name:'QQ'}
        ]
    )

1.2.2 删:remove

语法: db.collection.remove(查询表达式, 选项);
选项是指  {justOne:true/false},是否只删一行, 默认为false
​
注意
1: 查询表达式依然是个json对象
2: 查询表达式匹配的行,将被删掉
3: 如果不写查询表达式,collections中的所有文档将被删掉
​
例1: db.stu.remove({sn:'001'});
删除stu表中 sn属性值为'001'的文档 
​
例2: db.stu.remove({gender:'m',true});
删除stu表中gender属性为m的文档,只删除1行

1.2.3 改:update

改谁? --- 查询表达式
改成什么样? -- 新值 或 赋值表达式
操作选项 ----- 可选参数
语法: db.collection.update(查询表达式,新值,选项);
​
例1 :
db.news.update({name:'QQ'},{name:'MSN'});
​
是指选中news表中,name值为QQ的文档,并把其文档值改为{name:'MSN'},
​
结果: 文档中的其他列也不见了,改后只有_id和name列了.
​
即--新文档直接替换了旧文档,而不是修改
​
 
​
例2:如果是想修改文档的某列,可以用$set关键字
    db.collectionName.update(query,{$set:{name:'QQ'}})
​
修改时的赋值表达式
    $set  修改某列的值
    $unset 删除某个列
    $rename 重命名某个列
    $inc 增长某个列
    $setOnInsert 当upsert为true时,并且发生了insert操作时,可以补充的字段
​
Option的作用:
    {upsert:true/false,multi:true/false}
    upsert---是指没有匹配的行,则直接插入该行.(和mysql中的replace一样)
​
例3:
    db.stu.update({name:'wuyong'},{$set:{name:'junshiwuyong'}},{upsert:true});
如果有name='wuyong'的文档,将被修改,如果没有,将添加此新文档
​
例4:
    db.news.update({_id:99},{x:123,y:234},{upsert:true});
没有_id=99的文档被修改,因此直接插入该文档
​
multi: 是指修改多行(即使查询表达式命中多行,默认也只改1行,如果想改多行,可以用此选项)
例5:
    db.news.update({age:21},{$set:{age:22}},{multi:true});
则把news中所有age=21的文档,都修改

1.2.4 查:find, findOne

语法: db.collection.find(查询表达式,查询的列);
    db.collections.find(表达式,{列1:1,列2:1});
​
例1:查询所有文档 所有内容
    db.stu.find()
​
例2:查询所有文档,的gender属性 (_id属性默认总是查出来)
    db.stu.find({},{gendre:1})
​
例3: 查询所有文档的gender属性,且不查询_id属性
    db.stu.find({},{gender:1, _id:0})
​
例4: 查询所有gender属性值为male的文档中的name属性
    db.stu.find({gender:'male'},{name:1,_id:0});

1.3 查询表达式

1: 最简单的查询表达式
    {filed:value} ,是指查询field列的值为value的文档
2: $ne --- != 查询表达式,作用--查filed列的值 不等于 value 的文档
    {field:{$nq:value}}
3: $nin --> not in
4: $all
    语法: {field:{$all:[v1,v2..]}} ,是指取出 field列是一个数组,且至少包含 v1,v2值
5: $exists 
    语法: {field:{$exists:1}} 作用: 查询出含有field字段的文档
6: $nor
    {$nor,[条件1,条件2]}    是指  所有条件都不满足的文档为真返回
7: 用正则表达式查询 以”诺基亚”开头的商品
    例:db.goods.find({goods_name:/诺基亚.*/},{goods_name:1});
8: 用$where表达式来查询
    例: db.goods.find({$where:'this.cat_id != 3 && this.cat_id != 11'});
注意: 用$where查询时, mongodb是把bson结构的二进制数据转换为json结构的对象,然后比较对象的属性是否满足表达式,速度较慢
​
Update时可用的操作符
例:
->db.user.insert({name:'lisi',age:12,sex:'male',height:123,area:'haidian'});
->db.user.update({name:'lisi'},{$set:{area:'chaoyang'},$unset:{height:1},$inc:{age:1},$rename:{sex:'gender'}});
​
-> db.user.find();
​
{ "_id" : ObjectId("51fc01c4f5de93e1f2856e33"), "age" : 13, "area" : "chaoyang", "gender" : "male", "name" : "lisi" }
​
$setOnInsert ->相当于mysql中的列的默认值
​

1.4 游标cursor

游标是什么? 通俗的说,游标不是查询结果,而是查询的返回资源,或者接口.
通过这个接口,你可以逐条读取.就像php中的fopen打开文件,得到一个资源一样, 通过资源,可以一行一行的读文件.
​
声明游标:
var cursor =  db.collectioName.find(query,projection);
cursor.hasNext() ;  判断游标是否已经取到尽头
cursor. Next() ;    取出游标的下1个单元
​
用while来循环游标
    var mycursor = db.bar.find({_id:{$lte:5}})
    while(mycursor.hasNext()) {
        printjson(mycursor.next());
    }
    
例1:
// 声明游标
var cursor = db.goods.find();
// 循环游标
for(var doc=true;cursor.hasNext();) {
    printjson(cursor.next());
}
也可以简写:
for(var  cursor=db.goods.find(), doc=true;cursor.hasNext();) {
    printjson(cursor.next());
}
游标还有一个迭代函数,允许我们自定义回调函数来逐个处理每个单元.
cursor.forEach(回调函数);
​
例2:
var gettitle = function(obj) {
    print(obj.goods_name)
}
var cursor = db.goods.find();
cursor.forEach(gettitle);
​
游标在分页中的应用
    比如查到10000行,跳过100页,取10行.一般地,我们假设每页N行, 当前是page页就需要跳过前 (page-1)*N 行, 再取N行, 在mysql中, limit offset,N来实现
​
在mongo中,用skip(), limit()函数来实现的
    如 var mycursor = db.bar.find().skip(9995);则是查询结果中,跳过前9995行
​
查询第901页,每页10条
则是 var mytcursor = db.bar.find().skip(9000).limit(10);
​
通过cursor一次性得到所有数据, 并返回数组.
例:
    var cursor = db.goods.find();
    printjson(cursor.toArray()); //看到所有行
    printjson(cursor.toArray()[2]);  //看到第2行
注意: 不要随意使用toArray(),原因: 会把所有的行立即以对象形式组织在内存里.可以在取出少数几行时,用此功能.
​

1.5 索引创建

1、索引提高查询速度,降低写入速度,权衡常用的查询字段,不必在太多列上建索引

2、在mongodb中,索引可以按字段升序/降序来创建,便于排序

3、默认是用btree来组织索引文件,2.4版本以后,也允许建立hash索引

1.5.1 查看查询计划

db.find(query).explain();

"cursor" : "BasicCursor", ----说明没有索引发挥作用

"nscannedObjects" : 1000 ---理论上要扫描多少行

cursor" : "BtreeCursor sn_1", 用到的btree索引

1.5.2 常用命令

1、查看当前索引状态: 
	db.collection.getIndexes();
2、创建普通的单列索引:
	db.collection.ensureIndex({field:1/-1});  1是升续 2是降续
3、删除单个索引
	db.collection.dropIndex({filed:1/-1});
4、删除所有索引
	db.collection.dropIndexes();
5、创建多列索引  
	db.collection.ensureIndex({field1:1/-1, field2:1/-1});
6、创建子文档索引
	db.collection.ensureIndex({filed.subfield:1/-1});
7、创建唯一索引
	db.collection.ensureIndex({filed.subfield:1/-1}, {unique:true});
8、创建稀疏索引
	稀疏索引的特点------如果针对field做索引,针对不含field列的文档,将不建立索引.
9、与之相对,普通索引,会把该文档的field列的值认为NULL,并建索引.
适宜于: 小部分文档含有某列时.
	db.collection.ensureIndex({field:1/-1},{sparse:true});

-> db.tea.find();
{ "_id" : ObjectId("5275f99b87437c610023597b"), "email" : "a@163.com" }
{ "_id" : ObjectId("5275f99e87437c610023597c"), "email" : "b@163.com" }
{ "_id" : ObjectId("5275f9e887437c610023597e"), "email" : "c@163.com" }
{ "_id" : ObjectId("5275fa3887437c6100235980") }

如上内容,最后一行没有email列,
如果分别加普通索引,和稀疏索引,对于最后一行的email分别当成null 和 忽略最后一行来处理.
根据{email:null}来查询,前者能查到,而稀疏索引查不到最后一行.

创建哈希索引(2.4新增的),哈希索引速度比普通索引快,但是,无能对范围查询进行优化.适宜于---随机性强的散列
	db.collection.ensureIndex({file:'hashed'});

重建索引
一个表经过很多次修改后,导致表的文件产生空洞,索引文件也如此.可以通过索引的重建,减少索引文件碎片,并提高索引的效率.类似mysql中的optimize table
	db.collection.reIndex() 

1.6 Mongodb导出与导入

导入/导出可以操作的是本地的mongodb服务器,也可以是远程的.所以,都有如下通用选项

-h host 主机

--port port 端口

-u username 用户名

-p passwd 密码

1.6.1 导出-mongoexport

导出哪个库,哪张表,哪几列,哪几行?
    -d  库名
    -c  表名
    -f  field1,field2...列名
    -q  查询条件
    -o  导出的文件名
    -- csv  导出csv格式(便于和传统数据库交换数据)
例1: 查询test库 news表,导出为test.json文件夹到当前目录
    ./bin/mongoexport -d test -c news -o test.json
    more test.json
{ "_id" : { "$oid" : "51fc59c9fecc28d8316cfc03" }, "title" : "aaaa" }
{ "_id" : { "$oid" : "51fcaa3c5eed52c903a91837" }, "title" : "today is sataday" }
{ "_id" : { "$oid" : "51fcaa445eed52c903a91838" }, "title" : "ok now" }

例2: 只导出goods_id,goods_name列
	./bin/mongoexport -d test -c goods -f goods_id,goods_name -o goods.json

例3: 只导出价格低于1000元的行
	./bin/mongoexport -d test -c goods -f goods_id,goods_name,shop_price -q '{shop_price:{$lt:200}}' -o goods.json

注: _id列总是导出

1.6.2 导入-mongoimport

-d 待导入的数据库
-c 待导入的表(不存在会自己创建)
--type  csv/json(默认)
--file 备份文件路径

例1: 导入json
	./bin/mongoimport -d test -c goods --file ./goodsall.json
例2: 导入csv
	./bin/mongoimport -d test -c goods --type csv -f goods_id,goods_name --file ./goodsall.csv 

	./bin/mongoimport -d test -c goods --type csv --headline -f goods_id,goods_name --file ./goodsall.csv 

1.6.3 mongodump-mongorestore

mongodump 导出二进制bson结构的数据及其索引信息

-d  库名
-c  表名
-f  field1,field2...列名
例: 
	mongodum -d test  [-c 表名]  默认是导出到mongo下的dump目录
规律: 

1:导出的文件放在以database命名的目录下
2: 每个表导出2个文件,分别是bson结构的数据文件, json的索引信息
3: 如果不声明表名, 导出所有的表

mongorestore 导入二进制文件

例:
 ./bin/mongorestore -d test --directoryperdb dump/test/ (mongodump时的备份目录)

二进制备份,不仅可以备份数据,还可以备份索引, 备份数据比较小.

1.7 mongodb的用户管理

A) 在mongodb中,有一个admin数据库, 牵涉到服务器配置层面的操作,需要先切换到admin数据.
    即 use admin , -->相当于进入超级用户管理模式.
B) mongo的用户是以数据库为单位来建立的, 每个数据库有自己的管理员.
C) 我们在设置用户时,需要先在admin数据库下建立管理员---这个管理员登陆后,相当于超级管理员.
​
例: 添加用户
    命令:db.addUser();
简单参数: db.addUser(用户名,密码,是否只读)
​
注意: 添加用户后,我们再次退出并登陆,发现依然可以直接读数据库?原因: mongodb服务器启动时, 默认不是需要认证的.
要让用户生效, 需要启动服务器时,就指定 --auth 选项.这样, 操作时,就需要认证了.
​
例1: 添加用户
    -> use admin
    -> db.addUser('sa','sa',false);
例2: 认证
    -> use test
    -> db.auth(用户名,密码);
例3: 修改用户密码
    -> use test
    -> db.changeUserPassword(用户名, 新密码);
例4:删除用户
    -> use test
    -> db.removeUser(用户名);
注: 如果需要给用户添加更多的权限,可以用json结构来传递用户参数
例:
    -> use test
    ->db.addUser({user:'guan',pwd:'111111',roles:['readWrite,dbAdmin']});
​

1.8 mongodb的性能监测工具

mongotop 观察N秒内,每个库上读写花费的时间

Mongostat

 

inserts 每秒插入
query 每秒查询
update 每秒更新
delete 每秒删除
getmore 每秒查询游标
command 每秒总命令
flushes 每秒同步次数
mapped mmap内存大小(M)
size 虚拟内存(M)
res 物理内存(M)
faults 取内存页面失败次数(要去swap调)
locked db 锁住某库的时间
idx miss 索引未命中率
qr 队列里读取命令
qw 队列里写入命令
ar 活动的读取命令
aw 活动的读取命令
netIn 接收的流量
netOut 发出的流量
conn 当前打开连接数
set 复制集名称
repl 在复制集中的名称
​
db.serverStatus();
http://IP:mongod端口+1000
例: http://127.0.0.1:28017

1.9 sql语法示例

1、#查询每个栏目下的商品数量
{
    key:{cat_id:1},
    cond:{},
    reduce:function(curr,result) {
      result.cnt += 1;
    },
    initial:{cnt:0}
}
​
2、#查询每个栏目下价格高于50元的商品数量
{
    key:{cat_id:1},
    cond:{shop_price:{$gt:50}},
    reduce:function(curr,result) {
      result.cnt += 1;
    },
    initial:{cnt:0}
}
​
3、#每个栏目下的商品库存量 sum()操作
{
    key:{cat_id:1},
    cond:{},
    reduce:function(curr,result) {
      result.num += curr.goods_number;
    },
    initial:{num:0}
}
​
4、#查询每个栏目最贵的商品价格, max()操作
{
    key:{cat_id:1},
    cond:{},
    reduce:function(curr , result) {
      if(curr.shop_price > result.max) {
        result.max = curr.shop_price;
      }
    },
    initial:{max:0}
}
​
5、#同上,查询每个栏目下最便宜的商品价格,同学们自行完成
​
6、#查询每个栏目下商品的平均价格
{
    key:{cat_id:1},
    cond:{},
    reduce:function(curr , result) {
      result.cnt += 1;
      result.sum += curr.shop_price;
    },
    initial:{sum:0,cnt:0},
    finalize:function(result) {
      result.avg = result.sum/result.cnt;
    }
}
​
7、#查询每个栏目下的商品数量
db.collection.aggregate();
[
    {$group:{_id:"$cat_id",total:{$sum:1}}}
]
​
8、#查询goods下有多少条商品,select count(*) from goods
[
    {$group:{_id:null,total:{$sum:1}}}
]
​
9、#查询每个栏目下 价格大于50元的商品个数
[
    {$match:{shop_price:{$gt:50}}},
    {$group:{_id:"$cat_id",total:{$sum:1}}}
]
​
10、#查询每个栏目下 价格大于50元的商品个数,并筛选出"满足条件的商品个数" 大于等于3的栏目 
[
    {$match:{shop_price:{$gt:50}}},
    {$group:{_id:"$cat_id",total:{$sum:1}}},
    {$match:{total:{$gte:3}}}
]
​
11、#查询每个栏目下的库存量
[
    {$group:{_id:"$cat_id" , total:{$sum:"$goods_number"}}}
]
​
12、#查询每个栏目下的库存量,并按库存量排序
[
    {$group:{_id:"$cat_id" , total:{$sum:"$goods_number"}}},
    {$sort:{total:1}}
]
​
13、#查询每个栏目下的库存量,并按库存量排序
[
    {$group:{_id:"$cat_id" , total:{$sum:"$goods_number"}}},
    {$sort:{total:1}},
    {$limit:3}
]
​
14、#查询每个栏目的商品平均价格,并按平均价格由高到低排序
[
{$group:{_id:"$cat_id" , avg:{$avg:"$shop_price"}}},
{$sort:{avg:-1}}
]
​
mapReduce 随着"大数据"概念而流行.其实mapReduce的概念非常简单,从功能上说,相当于RDBMS的 group 操作
mapReduce的真正强项在哪?
答:在于分布式,当数据非常大时,像google,有N多数据中心,数据都不在地球的一端,用group力所不及.
group既然不支持分布式,单台服务器的运算能力必然是有限的.而mapRecuce支持分布式,支持大量的服务器同时工作,用蛮力来统计.
​
mapReduce的工作过程:
    map-->映射
    reduce->归约
    map: 先是把属于同一个组的数据,映射到一个数组上.cat_id-3 [23,2,6,7]
    reduce: 把数组(同一组)的数据,进行运算.
    用mapReduce计算每个栏目的库存总量
map函数
var map = function() {
  emit(this.cat_id,this.goods_number);
}
var reduce = function(cat_id,numbers) {
  return Array.sum(numbers);
}
db.goods.mapReduce(map,reduce,{out:'res'});
​
15、#用mapReduce计算每个栏目下商品的平均价格
var map = function() {
  emit(this.cat_id,this.shop_price);
}
var reduce = function(cat_id,values) {
  return Array.avg(values);
}
db.goods.mapReduce(map,reduce,{out:'res'});
​
  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

步道师就是我

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值