1、mongodb简介
-
MongoDB 是由C++语言编写的,是一个基于分布式文件存储的开源数据库系统。
-
在高负载的情况下,添加更多的节点,可以保证服务器性能。
-
MongoDB 旨在为WEB应用提供可扩展的高性能数据存储解决方案。
-
MongoDB 将数据存储为一个文档,数据结构由键值(key=>value)对组成。MongoDB 文档类似于 JSON 对象。字段值可以包含其他文档,数组及文档数组

2、mongodb下载地址
www.mongodb.com/download-ce… 选择适合自己的版本。

- 需要注意的是,例如windows下,安装完成需要把环境变量设置到你安装mongodb文件夹的bin目录下。
- 比如我安装mongodb到E:\mongodb所以bin目录就是E:\mongodb\bin
3、mongodb 的适用场景和基本概念
3.1 MongoDB适用场景
-
网站数据、缓存等大尺寸、低价值的数据
-
在高伸缩性的场景,用于对象及JSON数据的存储。
3.2 MongoDB的基本概念
数据库
MongoDB的一个实例可以拥有一个或多个相互独立的数据库,每个数据库都有自己的集合。
集合
集合可以看作是拥有动态模式的表。
文档
文档是MongoDB中基本的数据单元,类似于RDB的行。文档是键值对的一个有序集合。在JS中,文档被表示成对象。
_id
每个文档都有个特殊的“_id”,在文档所属集合中是唯一的。
JavaScript shell
MongoDB自带了一个功能强大的JavaScript Shell,可以用于管理或操作MongoDB
举例
- 数据库 MongoDB的单个实例可以容纳多个独立的数据库,比如一个学生管理系统就可以对应一个数据库实例
- 集合 数据库是由集合组成的,一个集合用来表示一个实体,如学生集合
- 文档 集合是由文档组成的,一个文档表示一条记录,比如一位同学张三就是一个文档
4、添加和删除服务
添加服务 在命令行工具里
mongod.exe --dbpath E:\mongodata --logpath E:\mongolog --logappend --directoryperdb --serviceName mongodb --install
- --dbpath 指定数据存放目录
- --logpath 日志放在哪里
- --logappend 日志写入方式,可以给老日志加
- --directoryperdb 将每个数据库单独存放
- --serviceName 起的服务名字叫
- --install 安装服务
下图表示添加服务成功了
5、mongodb启动与连接
启动mongodb服务
mongod --dbpath=E:\Mongodb\data
如果出现waiting for connections on port 27017就表示启动成功,已经在27017端口上监听了客户端的请求注:启动说明
选项 | 含义 |
---|---|
--port | 指定服务端口号,默认端口27017 |
--logpath | 指定MongoDB日志文件,注意是指定文件不是目录 |
--logappend | 使用追加的方式写日志 |
--dbpath | 指定数据库路径 |
--directoryperdb | 设置每个数据库将被保存在一个单独的目录 |
接着,再打开一个新的命令行窗口,输入
mongo
就可以链接到数据库了
最后介绍一款mongodb可视化工具 Studio3T
6、数据库操作
6.1 使用数据库
use database_name
// 例如我们这里可以建一个叫school的数据库
use school
database_name 代表数据库的名字
注:如果此数据库存在,则切换到此数据库下,如果此数据库还不存在也可以切过来,但是并不能立刻创建数据库 切换到 school数据库下

6.2 查看所有数据库
show dbs
备注: 我们刚创建的数据库school如果不在列表内, 要显示它,我们需要向school数据库插入一些数据 db.students.insert({age:1});
inert数据之后

6.3 查看当前使用的数据库
db
注:db代表的是当前数据库 也就是school这个数据库

6.4 删除数据库
db.dropDatabase()
school数据库就没了

7、集合操作
7.1 查看集合帮助
use demo
db.demo.help()

7.2 创建集合
7.2.1 创建一个空集合
db.createCollection(collection_Name)
collection_Name集合的名称

7.2.2 创建集合并插入一个文档
collection_Name集合的名称 document要插入的文档
db.collection_Name.insert(document)
// 例如
db.demo.insert({age:1})

之前我们使用MySQL等关系型数据库时,主键都是设置成自增的。但在分布式环境下,这种方法就不可行了,会产生冲突。为此,MongoDB采用了一个称之为ObjectId的类型来做主键。ObjectId是一个12字节的 BSON 类型字符串。按照字节顺序,一次代表:
- | - |
---|---|
4字节: | UNIX时间戳 |
3字节: | 表示运行MongoDB的机器 |
2字节: | 表示生成此_id的进程 |
3字节: | 由一个随机数开始的计数器生成的值 |
7.3 查看数据库下的集合
show collections

7.4 查看当前集合
db.getName() 或者 db
7.5 删除当前集合
db.dropDatabase()

8. 插入文档
8.1 insert
db.collection_name.insert(document);
collection_name 集合的名字
document 插入的文档
注:每当插入一条新文档的时候mongodb会自动为此文档生成一个_id属性,_id一定是唯一的,用来唯一标识一个文档 _id也可以直接指定,但如果数据库中此集合下已经有此_id的话插入会失败
db.demo.insert({_id:1,age:1});


8.2 save
db.collection_name.save(document)
collection_name 集合的名字 document 插入的文档
注:如果不指定 _id 字段 save() 方法类似于 insert() 方法。如果指定 _id 字段,则会更新该 _id 的数据。 我们save之前保存过的id=1的数据,看是不是更新了此条数据
db.demo.save({_id:1, age:2})

9. 更新文档
db.collection.update(
<query>,
<updateObj>,
{
upsert: <boolean>,
multi: <boolean>
}
)
参数 | - |
---|---|
query | 查询条件,指定要更新符合哪些条件的文档 |
update | 更新后的对象或指定一些更新的操作符 |
$set | 在原基础上累加 |
upsert | 可选,这个参数的意思是,如果不存在符合条件的记录时是否插入updateObj. 默认是false,不插入。 |
multi | 可选,mongodb 默认只更新找到的第一条记录,如果这个参数为true,就更新所有符合条件的记录。 |
9.1 更新文档
db.demo.insert({name: 'zhangsan'});
db.demo.update({name:'zhangsan'},{age:12})

9.2 upsert
将students集合中数据中name是lisi的值修改为lisi,因为原表根本不存在name:lisi的,原本不会插入这一条数据,但是upsert选项设为了true就插入成功了
db.demo.update({name:'lisi'}, {name:'lisi'}, {upsert:true});

9.3 multi
如果有多条age是1的数据只更新一条,如果想全部更新需要指定{multi:true}的参数
db.school.insert({age:1});
db.school.insert({age:1});
db.school.insert({age:1});
db.school.update({age:1}, {$set:{name:'lisi'}}, {multi:true});
db.school.find({})

10、更新操作符
10.1 $set
直接指定更新后的值
use c3;
db.c3.insert({name:'a'});
db.c3.update({name:'a'},{$set:{age:10}});
db.c3.find({})

10.2 $inc
在原基础上累加
db.c3.update({name:'a'},{$inc:{age:1}});

10.3 $unset
删除指定的键
db.c3.update({name:'2'},{$unset:{age:11}});

10.4 $push
向数组中添加元素
var result = db.student.update({name:'张三'},{
$push:{"hobbys":"smoking"}
});

10.5 $ne
$ne类似于MYSQL的 not in 或者not exists
db.c3.update({hobbys:{$ne:"eating"}},{$push:{"hobbys":"sleeping"}});

10.6 $addToSet
向集合中添加元素
db.c3.update({a:1},{$addToSet:{"hobbys":"smoking"}});

10.7 $each
把数组中的元素逐个添加到集合中
db.a.insert({name:'a', hobbys:[1,2]});
db.a.update({name:'a'},{$addToSet:{hobbys:{$each:[3,4]}}});
db.a.find()

10.8 $pop
从数组中移除指定的索引中对应的元素
- {$pop:{"key":1}}从数组末尾删除一个与元素
- {$pop:{"key":-1}}从数组开头删除一个与元素
db.c3.update({name:'a'},{$pop:{hobbys:1}});
db.c3.find()

10.9 修改指定索引元素
db.c3.update({name:'zfpx1'},{$set:{"hobbys.0":"smoking2"}});

11、文档的删除
remove方法是用来移除集合中的数据
11.1 语法
db.collection.remove(
<query>,
{
justOne: <boolean>
}
)
11.2 参数
- | - |
---|---|
query | (可选)删除的文档的条件。 |
justOne | (可选)如果设为 true 或 1,则只删除匹配到的多个文档中的第一个 |
11.3 实例
删除worker集合里name是zfpx2的所有文档数据
db.c3.remove({name:'a'});

db.c3.remove({name:'a'},{justOne:true});
12、查询文档
12.1 find
语法
db.collection_name.find()
参数
- | - |
---|---|
collection_name | 集合的名字 |
实例
查询c3下所有的文档
db.students.find()
12.2 查询指定列
语法
db.collection_name.find({queryWhere},{key:1,key:1})
参数列表
- | - |
---|---|
collection_name | 集合的名字 |
queryWhere | 参阅查询条件操作符 |
key | 指定要返回的列 |
1 | 表示要显示 |
实例
只返回显示age列
db.c3.insert([{name:1,age:2},{name:1,age:3},{name:1,age:4}]);
db.c3.find({},{_id:0, age:1});

12.3 findOne
查询匹配结果的第一条数据 语法
db.collection_name.findOne()
实例
db.c3.findOne()`
12.4 $in
查询字段在某个范围内
db.c3.insert([{age:30, name:1},{age:100, name:2},{age:40, name:3},{age:60, name:4}])
db.c3.find({age:{$in:[30,100]}},{name:1,age:1});

12.5 $nin
查询字段不在某个范围内
db.c3.find({age:{$nin:[30,100]}},{name:1,age:1});

12.6 $not
对特定条件取反
db.c3.find({age:{$not:{$gte:20,$lte:30}}});

12.7 array
对数组的查询
// 按所有元素匹配
db.c3.find({friends:[ "A", "B", "C", "D" ]});
//匹配一项 包含A的就可以
db.c3.find({friends:"A"});
// $all 必须同时包含A B
db.c3.find({friends:{$all:['A',"B"]}});
// $in 或者关系 ,包含A或者B
db.c3.find({friends:{$in:['A',"B"]}});
//$size 按数组的长度去匹配
db.c3.find({friends:{$size:4}});
//$slice 只返回数组中的某一部分
db.c3.find({friends:{$size:5}},{name:1,friends:{$slice:2}});
db.c3.find({friends:{$size:5}},{name:1,friends:{$slice:-2}});
$slice:["$array", [startIndex, ] length ] (startIndex可以省略,默认从0开始)
db.c3.find({},{friends:{$slice:[0,3]}});
12.8 where
db.c3.find({$where:"this.age>30"},{name:1,age:1});

12.9 cursor
游标不是查询结果,而是查询的一个返回资源或者接口,通过这个接口,可以逐条读取数据
var result = db.student.find();
while(result.hasNext()){
printjson(result.next());
}
13、条件操作符
条件操作符用于比较两个表达式并从mongoDB集合中获取数据
13.1 大于、大于等于、小于、小于等于操作符
参数
- | - |
---|---|
$gt | 大于 |
$gte | 大于等于 |
$lt | 小于 |
$lte | 小于等于 |
13.2 使用 _id进行查询
语法
db.collectoin_name.find({"_id" : ObjectId("value")})
查询_id是 562af23062d5a57609133974 数据
db.stuc3dents.find({_id:ObjectId("5adb666ecd738e9771638985")});
13.8 查询结果集的条数
语法
db.collectoin_name.find().count();
db.c3.find().count()

13.9 正则匹配
语法
db.collection.find({key:/value/})
参数
- | - |
---|---|
collectoin_name | 集合名称 |
key | 字段 |
value | 值 |
查询name里包含joe的数据,正则只能匹配字符串
db.user.find({name:/joe/})
14、与和或
14.1 and
find方法可以传入多个键(key),每个键(key)以逗号隔开
语法
db.collection_name.find({key1:value1, key2:value2})
查询name是1并且age是30的数据
db.students.find({name:1,age:30})

14.2 or
语法
db.collection_name.find(
{
$or: [
{key1: value1}, {key2:value2}
]
}
)
查询age = 30 或者 age = 100 的数据
db.c3.find({$or:[{age:30},{age:100}]})

14.3 and和or联用
语法
db.collection_name.find(
{
key1:value1,
key2:value2,
$or: [
{key1: value1},
{key2:value2}
]
}
)
实例 查询 age是30 并且 name是1或者 name是 3 的数据
db.c3.find({age: 30, $or:[{name:1},{age:3}]})

15、分页查询
15.1 limit
读取指定数量的数据记录 语法
db.collectoin_name.find().limit(number)
var a = [];
for(var i = 0;i<10;i++){
a.push({age:i})
};
db.c3.insert(a);
db.students.find().limit(3)

15.2 skip
跳过指定数量的数据,skip方法同样接受一个数字参数作为跳过的记录条数 语法
db.collectoin_name.find().skip(number)
db.c3.find().skip(3)

15.3 skip+limit
通常用这种方式来实现分页功能 语法
db.collectoin_name.find().skip(skipNum).limit(limitNum)
db.c3.find().skip(3).limit(3);

15.4 sort排序
sort()方法可以通过参数指定排序的字段,并使用 1 和 -1 来指定排序的方式,其中 1 为升序排列,而-1是用于降序排列。 语法
db.collectoin_name.find().sort({key:1})
db.collectoin_name.find().sort({key:-1})
db.c3
.find().sort({age:1})