操作mongodb数据库
【操作mongodb数据库】
简介
- 1.Mongoose是一个让我们可以通过Node来操作MongoDB的模块。
- 2.Mongoose是一个对象文档模型(ODM)库,它对Node原生的MongoDB模块进行了进一步的优化封装,并提供了更多的功能。在大多数情况下,它被用来把结构化的模式应用到一个MongoDB集合,并提供了验证和类型转换等好处
- 3.mongoose中的对象:
- Schema 模式对象(Schema对象定义约束了数据库中的文档结构)
- Model 模型对象(Model对象作为集合中的所有文档的表示,相当于MongoDB数据库中的集合collection)
- Document 文档对象(Document表示集合中的具体文档,相当于集合中的一个具体的文档)
mongoose的好处
- 可以为文档创建一个模式结构(Schema)
- 可以对模型中的对象/文档进行验证
- 数据可以通过类型转换转换为对象模型
- 可以使用中间件来应用业务逻辑挂钩
- 比Node原生的MongoDB驱动更容易
安装
1 | npm i -S mongoose |
连接数据库
config/db.config.js
1 | // 1.引入mongoose |
注:MongoDB数据库,一般情况下,只需要连接一次,连接一次后,除非项目停止服务器关闭,否则连接一般不会断开
在bin目录下的www文件中使用直接require(“…/config/db.config.js”)进行数据库连接的启动
创建模式对象和模型对象
数据库中的 Schema,为数据库对象的集合。schema 是 mongoose 里会用到的一种数据模式,可以理解为表结构的定义;每个 schema会映射到 mongodb 中的一个 collection,它不具备操作数据库的能力。
- 每个 schema 都会映射到一个 MongoDB collection 并定义这个collection里的文档结构
- 支持的字段类型
类型 | 作用 |
---|---|
String | 定义字符串 |
Number | 定义数字 |
Date | 定义日期 |
Buffer | 定义二进制 |
Boolean | 定义布尔值 |
Mixed | 定义混合类型 |
ObjectId | 定义对象ID |
Array | 定义数组 |
model/UserModel.js
1 | const mongoose = require("mongoose") |
文档新增
save()
- 操作的是文档
案例:
1 | var mongoose = require('mongoose') |
create()
-
操作模型
-
Model.create(doc(s), [callback])
-
参数:
[doc(s)]:文档对象或文档对象数组
[callback]:回调函数
1 | var mongoose = require('mongoose') |
其它:
1 | //Model.createOne(doc, [callback]); 创建一个对象 |
insertMany()
- Model.insertMany(doc(s), [options], [callback])
- 返回值为一个数组
- 案例:
1 | UserModel.insertMany({name:"小明",age:18},{name:"小芳",age:14},(err,docs) => { |
文档查询
_id | name | grades | __v |
---|---|---|---|
6017befb5c36d64d08b72576 | 小明 | 68 | 0 |
6017befb5c36d64d08b72577 | 小芳 | 94 | 0 |
6017c455ba09d355a49ec8eb | 小红 | 52 | 0 |
6017c455ba09d355a49ec8ec | 小刚 | 46 | 0 |
find()
-
Model.find(conditions, [projection], [options], [callback])
-
参数
conditions:查询条件
[projection]:控制返回字段
[options]:配置查询参数
[callback]:回调函数–function(err,docs){}
-
案例:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53var mongoose = require('mongoose')
mongoose.connect('mongodb://localhost:27017/student',(err) => {
if(!err){
var schema = new mongoose.Schema({name:String,grades:Number})
var stuModel = mongoose.model('grades',schema)
//查询所有数据
stuModel.find((err,docs) => {
if(!err){
console.log(docs)
}
})
/* [{ _id: 6017befb5c36d64d08b72576, name: '小明', grades: 68, __v: 0 },
{ _id: 6017befb5c36d64d08b72577, name: '小芳', grades: 94, __v: 0 },
{ _id: 6017c455ba09d355a49ec8eb, name: '小红', grades: 52, __v: 0 },
{ _id: 6017c455ba09d355a49ec8ec, name: '小刚', grades: 46, __v: 0 }]*/
//查询成绩大于60以上的数据
stuModel.find({grades:{$gte:60}},(err,docs) => {
if(!err){
console.log(docs)
}
})
/*[{ _id: 6017befb5c36d64d08b72576, name: '小明', grades: 68, __v: 0 },
{ _id: 6017befb5c36d64d08b72577, name: '小芳', grades: 94, __v: 0 }]*/
//查询成绩大于60以上且名字里存在‘芳’的数据
stuModel.find({name:/芳/,grades:{$gte:60}},(err,docs) => {
if(!err){
console.log(docs)
}
})
/*[
* { _id: 6017befb5c36d64d08b72577, name: '小芳', grades: 94, __v: 0 }
* ]*/
//查询名字里存在‘明’的数据且只输出‘name’字段
//_id默认会返回
stuModel.find({name:/明/},{name:1,_id:0},(err,docs) => {
if(!err){
console.log(docs)
}
})
// [{name: '小明'}]
//跳过前两条数据并限制只输出一条数据
stuModel.find(null,null,{skip:2,limit: 1},(err,docs) => {
if(!err){
console.log(docs)
}
})
/*[{ _id: 6017c455ba09d355a49ec8eb, name: '小红', grades: 52, __v: 0 }*/
}
})
findById()
- Model.findById(id, [projection], [options], [callback])
- 案例:
1 | var mongoose = require('mongoose') |
findOne()
- 返回查询到的数据的第一个
- Model.findOne([conditions], [projection], [options], [callback])
- 案例:
1 | var mongoose = require('mongoose') |
复杂查询【$where】
-
$where 可以使用任意的 JavaScript 作为查询的一部分,包含JavaScript 表达式的字符串或者函数
-
案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32var mongoose = require('mongoose')
mongoose.connect('mongodb://localhost:27017/student',(err) => {
if(!err){
var schema = new mongoose.Schema({name:String,grades:Number})
//添加一个测试字段
// schema.add({test:Number})
var stuModel = mongoose.model('grades',schema)
//添加两条数据
// stuModel.create({name:"小花",grades:76,test:76},{name:"小兰",grades:60,test:30},(err,docs)=>{
// console.log(docs)
// })
//字符串 es5中this与obj指向一样,es6中只能用obj
stuModel.find({$where:"this.grades == this.test" || "obj.grades == obj.test"},(err,doc) => {
if(!err){
console.log(doc)
}
})
//[{_id: 6017d7cb8a95cb2a00aae3ae,name: '小花',grades: 76,test: 76,__v: 0}]
//函数
stuModel.find({$where:function() {
return this.grades == this.test || obj.grades == obj.test*2
}},(err,doc) => {
if(!err){
console.log(doc)
}
})
/*[{_id: 6017d7cb8a95cb2a00aae3ae,name: '小花',grades: 76,test: 76,__v: 0},
{_id: 6017d7cb8a95cb2a00aae3af,name: '小兰',grades: 60,test: 30,__v: 0}]*/
}
})
常用查询条件
1 | $or 或关系 |
特定类型查询
_id | name | grades | __v | test |
---|---|---|---|---|
6017befb5c36d64d08b72576 | 小明 | 68 | 0 | 1 |
6017befb5c36d64d08b72577 | 小芳 | 94 | 0 | 3 |
6017c455ba09d355a49ec8eb | 小红 | 52 | 0 | 5 |
6017c455ba09d355a49ec8ec | 小刚 | 46 | 0 | 2 |
6017d7cb8a95cb2a00aae3ae | 小花 | 76 | 0 | 4 |
6017d7cb8a95cb2a00aae3af | 小兰 | 60 | 0 | 6 |
方法
方法 | 作用 |
---|---|
sort | 排序 |
skip | 跳过 |
limit | 限制 |
select | 显示字段 |
exect | 执行 |
count | 计数 |
distinct | 去重 |
exec()和 then()
两者返回的都是 promise对象
exec一般用于独立的动作一次性执行,
then则用于连续性的动作
从其方法名也可以区别它们的用法,exec就是执行的意思,then就是然后怎么怎么,
exec和then的参数是有所不同的,前者是 callback(err,doc),后者则是 resolved(doc),rejected(err)
案例:
1 | const mongoose = require('mongoose') |
文档更新
update()
-
Model.update(conditions, doc, [options], [callback])
-
参数
conditions:查询条件
doc:需要修改的数据(插入的数据)
[options]:控制选项
safe (boolean): 默认为true。安全模式。
upsert (boolean): 默认为false。如果不存在则创建新记录。
multi (boolean): 默认为false。是否更新多个查询记录。
runValidators: 如果值为true,执行Validation验证。
setDefaultsOnInsert: 如果upsert选项为true,在新建时插入文档定义的默认值。
strict (boolean): 以strict模式进行更新。
overwrite (boolean): 默认为false。禁用update-only模式,允许覆盖记录。 -
[callback]:回调函数
-
若设置了查询条件,当数据库不满足时默认什么也不发生
-
update() 方法中的回调函数不能省略,否则数据不会更新,当回调无有用信息时可以使用exec()简化
1
stuModel.update({name:'小明'},{$set:{test:34}}.exec())
-
案例
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21//第一步,引入mongoose
const mongoose = require('mongoose')
//第二步,连接数据库
mongoose.connect('mongodb://localhost:27017/student',err=>{
if(!err){
//第三步,创建模板
var Schema =new mongoose.Schema({ name:String,grades:Number,test:{type:Number,default:0}})
// var Schema = new Schema()
//第四步,将模板映射到集合并创建
var stuModel = mongoose.model('grades',Schema)
//查询name为小明的数据,并将其test更改为34
//若有多个文档,默认只更新第一个
stuModel.update({name:'小明'},{$set:{test:34}},(err,raw)=>{
console.log(raw)
})
//{ n: 1, nModified: 1, ok: 1 }
//6017befb5c36d64d08b72576 小明 68 0 34
}
})
updateOne()
-
Model.updateOne(conditions, doc, [options], [callback])
-
与update()相似,唯一区别为updateOne() 默认更新一个文档,即使设置{multi:true}也无法只更新一个文档
updateMany()
-
Model.updateMany(conditions, doc, [options], [callback])
-
与update()相似,唯一区别为updateMany() 默认更新多个文档,即使设置{multi:false}也无法只更新一个文档
find()+save()
用于复杂更新
1 | const mongoose = require('mongoose') |
findOne() + save()
- 用于复杂更新
- findOne()返回值为文档对象
1 | const mongoose = require('mongoose') |
fingOneAndUpdate()
Model.findOneAndUpdate([conditions], [update], [options], [callback])
findByIdAndUpdate()
Model.findByIdAndUpdate([conditions], [update], [options], [callback])
文档删除
deleteOne()
- 会删除符合条件的所有数据
- Model的deleteOne()
1 | const mongoose = require('mongoose') |
- 文档的deleteOne()
1 | const mongoose = require('mongoose') |
findOneAndRemove()
-
删除符合条件的一条数据
-
Model.findOneAndRemove(conditions, [options], [callback])
-
回调不可省略,但可以使用exec() 简写
1 | stuModel.findOneAndRemove({name:/差生/}).exec() |
findByIdAndRemove()
- 通过id删除数据(id是唯一的)
- Model.findByIdAndRemove(conditions, [options], [callback])
- 回调不可省略,但可以使用exec() 简写
前后钩子
-
前后钩子即 pre() 和 post() 方法(中间件)
-
中间件在schema上指定,类似静态方法或实例方法等
-
可以在执行以下操作时设置前后钩子
init
validate
save
remove
count
find
findOne
findOneAndRemove
findOneAndUpdate
insertMany
update -
【pre()】:在执行某些操作前执行
-
【post】:在执行某些操作前后执行,不可以使用next()
案例:
1 | const mongoose = require('mongoose') |
文档验证
- 保证保存文档时,可以按照Schema设置的字段进行设置
【required】:数据必填
1 | //将name设置为必填字段,如果没有name字段,文档将不被保存,且出现错误提示 |
【default】:默认值
1 | //设置age字段的默认值为18,如果不设置age字段,则会取默认值 |
【min】【max】:最小/大值
- 只适用于数字
1 | //将age的取值范围设置为[0,10]。如果age取值为20,文档将不被保存,且出现错误提示 |
【match】:正则匹配
- 只适用于字符串
1 | //将name的match设置为必须存在'01'字符。如果name不存在'01',文档将不被保存,且出现错误提示 |
【enum】:枚举匹配
- 只适用于字符串
1 | //将name的枚举取值设置为['zs','ls','ww'],如果name不在枚举范围内取值,文档将不被保存,且出现错误提示 |
【validate】:自定义匹配
- validate实际上是一个函数,函数的参数代表当前字段,返回true表示通过验证,返回false表示未通过验证
1 | //定义名字name的长度必须在4个字符以上 |