0. 前言:
Mongoose是NodeJS的驱动,不能作为其他语言的驱动。Mongoose有两个特点:
- 通过关系型数据库的思想来设计非关系型数据库
- 基于mongodb驱动,简化操作
Mongooose中,有三个比较重要的概念,分别是Schema、Model、Document。它们的关系是:Schema生成Model,Model创造Document,Model和Document都可对数据库操作造成影响,但Model比Document更具操作性。
Schema
用于定义数据库的结构。类似创建表时的数据定义(不仅仅可以定义文档的结构和属性,还可以定义文档的实例方法、静态模型方法、复合索引等),每个Schema
会映射到mongodb中的一个collection,Schema
不具备操作数据库的能力。
Model是由Schema编译而成的构造器,具有抽象属性和行为,可以对数据库进行增删查改。Model的每一个实例(instance)就是一个文档document。
Document是由Model创建的实体,它的操作也会影响数据库。
1. 安装
在此之前需要安装nodejs和mongodb,才能使用npm来安装mongoose
npm install mongoose
项目中可使用下面的命令安装mongoose并自动将其添加到依赖
npm i mongoose --save
2. 连接数据库
2.1 首先使用 require() 引入mongoose
var mongoose = require('mongoose');
2.2 使用 connect() 来连接mongodb数据库
mongoose.connect(url);
connect()最简单的使用方式,就是只要传入url参数即可,如下所示。
/*直接连接*/ mongoose.connect("mongodb://127.0.0.1/mongoose_test" ); /*传递用户名密码来连接*/ mongoose.connect('mongodb://username:password@host:port/database?options...');
注意:mongoose版本大于或等于4.11.0时,连接数据库会提示warning(不影响运行),避免的方法添加userMongoClient的设置,且属性值为true
mongoose.connect('mongodb://数据库的ip地址:端口号/数据库名', { useMongoClient: true});
connect()方法还接受一个选项对象options,该对象将传递给底层驱动程序。
mongoose.connect(uri, options);
可用选项如下所示,更多详见参考,options 所包含的所有选项优先于连接字符串中传递的选项
db -数据库设置 server -服务器设置 replset -副本集设置 user -用户名 pass -密码 auth -鉴权选项 mongos -连接多个数据库
如果要连接多个数据库,只需要设置多个url以,
隔开,同时设置mongos
为true
mongoose.connect('urlA,urlB,...', { mongos : true })
connect()函数还接受一个回调参数,来判断是否连接成功
mongoose.connect(uri, options, function(error) { });
var mongoose = require('mongoose'); mongoose.connect("mongodb://localhost/mongoose_test", function(err) { if(err){ console.log('连接失败'); }else{ console.log('连接成功'); } });
2.3 使用 disconnect() 断开数据库连接(一般不需要调用)
MongoDB数据库,一般情况下,只需要连接一次,连接一次以后,除非项目停止服务器关闭,否则连接一般不会断开
mongoose.disconnect();
2.4 监听MongoDB数据库的连接状态
在mongoose对象中,有一个属性叫做connection,该对象表示的就是数据库连接。通过监视该对象的状态,可以来监听数据库的连接与断开
数据库连接成功的事件
mongoose.connection.once("open",function(){});
数据库断开的事件
mongoose.connection.once("close",function(){});
3. Schema
Schema主要用于定义MongoDB中集合Collection里文档document的结构。
定义Schema需要指定字段名和类型,支持的类型包括以下9种
String 字符串
Number 数字
Date 日期
Buffer 二进制
Boolean 布尔值
Mixed 混合类型
ObjectId 对象ID
Array 数组
Decimal128 Decimal类型
通过mongoose.Schema来调用Schema,然后使用new方法来创建schema对象
var mongoose = require("mongoose"); mongoose.connect("mongodb://127.0.0.1/mongoose_test",{useMongoClient:true}); mongoose.connection.once("open",function () { console.log("数据库连接成功~~~"); }); //将mongoose.Schema 赋值给一个变量 var Schema = mongoose.Schema; //创建Schema(模式)对象 var stuSchema = new Schema({ name:String, age:Number, gender:{ type:String, default:"female" }, address:String });
[注意]创建Schema对象时,声明字段类型有两种方法,一种是首字母大写的字段类型,另一种是引号包含的小写字段类型
var mySchema = new Schema({title:String, author:String}); //或者 var mySchema = new Schema({title:'string', author:'string'});
如果需要在Schema定义后添加其他字段,可以使用add()方法
var MySchema = new Schema; MySchema.add({ name: 'string', color: 'string', price: 'number' });
4. Model & Document
模型Model是根据Schema编译出的构造器,或者称为类,通过Model可以实例化出文档对象document。文档document的创建和检索都需要通过模型Model来处理
mongoose.model();
[注意]一定要将model()方法的第一个参数和其返回值设置为相同的值,否则会出现不可预知的结果
Mongoose会将集合名称设置为模型名称的小写版。如果名称的最后一个字符是字母,则会变成复数;如果名称的最后一个字符是数字,则不变;如果模型名称为"MyModel",则集合名称为"mymodels";如果模型名称为"Model1",则集合名称为"model1"。
实例化文档document
var StuModel = mongoose.model("student" , stuSchema); //Document 和 集合中的文档一一对应 , Document是Model的实例 通过Model查询到结果都是Document //创建一个Document var stu = new StuModel({ name:"孙悟空", age:18, gender:"male", address:"花果山" });
文档保存
通过new StuModel()创建的文档stu,必须通过save()方法,才能将创建的文档保存到数据库的集合中,集合名称为模型名称的小写复数版
//回调函数是可选项,第一个参数为err,第二个参数为保存的文档对象 save(function (err, doc) {})
stu.save(function (err) { if (err) return handleError(err); })
也可减少步骤,直接向数据库插入文档
var StuModel = mongoose.model("student" , stuSchema); //向数据库中插入一个文档 //StuModel.create(doc, function(err){}); StuModel.create({ name:"白骨精", age:16, address:"白骨洞" },function (err) { if(!err){ console.log("插入成功~~~"); } });
结果
5. Model对象的方法
有了Model,我们就可以来对数据库进行增删改查的操作了。具体方法可参考官方文档。
5.1 增加方法
Model.create(doc(s), [callback]) - 用来创建一个或多个文档并添加到数据库中 - 参数: doc(s) 可以是一个文档对象,也可以是一个文档对象的数组 callback 当操作完成以后调用的回调函数
StuModel.create([ { name:"沙和尚", age:38, gender:"male", address:"流沙河" } ],function (err) { if(!err){ console.log(arguments); } });
控制台输出
5.2 修改方法
Model.update(conditions, doc, [options], [callback]) Model.updateMany(conditions, doc, [options], [callback]) Model.updateOne(conditions, doc, [options], [callback]) - 用来修改一个或多个文档 - 参数: conditions 查询条件 doc 修改后的对象 options 配置参数 callback 回调函数 Model.replaceOne(conditions, doc, [options], [callback])
StuModel.updateOne({name:"唐僧"},{$set:{age:20}},function (err) { if(!err){ console.log("修改成功"); } });
5.3 查询方法
Model.find(conditions, [projection], [options], [callback]) - 查询所有符合条件的文档 总会返回一个数组 Model.findById(id, [projection], [options], [callback]) - 根据文档的id属性查询文档 Model.findOne([conditions], [projection], [options], [callback]) - 查询符合条件的第一个文档 总会返回一个具体的文档对象(非数组) conditions 查询的条件 projection 投影 需要获取到的字段 - 两种方式 {name:1,_id:0} "name -_id" options 查询选项(skip limit) {skip:3 , limit:1} callback 回调函数,查询结果会通过回调函数返回 回调函数必须传,如果不传回调函数,压根不会查询
官方文档查询方法截图如下
下面分别演示几种不同的查询方法
//查询name为“唐僧”的 StuModel.find({name:"唐僧"},function (err , docs) { if(!err){ console.log(docs); } });
结果
//只查询name,且不查询_id (_id字段默认输出) StuModel.find({},{name:1 , _id:0},function (err , docs) { if(!err){ console.log(docs); } });
结果
//只查询文档的name,age值,查询时跳过3个,结果集限制只查询出一个 StuModel.find({},"name age -_id", {skip:3 , limit:1} , function (err , docs) { if(!err){ console.log(docs); } });
结果
StuModel.findOne({} , function (err , doc) { if(!err){ console.log(doc); } });
结果
StuModel.findById("59c4c3cf4e5483191467d392" , function (err , doc) { if(!err){ //console.log(doc); //通过find()查询的结果,返回的对象,就是Document,文档对象 //Document对象是Model的实例 console.log(doc instanceof StuModel); } });
结果
5.4 删除方法
Model.remove(conditions, [callback])
Model.deleteOne(conditions, [callback])
Model.deleteMany(conditions, [callback])
StuModel.remove({name:"白骨精"},function (err) { if(!err){ console.log("删除成功~~"); } });
5.5 统计
Model.count(conditions, [callback])
- 统计文档的数量的
StuModel.count({},function (err , count) { if(!err){ console.log(count); } });
6. Document对象的方法
Document 和 集合中的文档一一对应,Document是Model的实例。通过Model查询到结果都是Document。
6.1 保存
save([options], [options.safe], [options.validateBeforeSave], [fn])
//创建一个Document var stu = new StuModel({ name:"奔波霸", age:48, gender:"male", address:"碧波潭" }); //保存改document对象 stu.save(function (err,product,numAffected) { //返回三个对象 err:失败信息; product:保存的对象; numAffected:该document对象被持久化了为1,否则为0 if(!err){ console.log("保存成功~~~"); }
6.2 更新
update(update,[options],[callback])
doc.update({$set:{age:28}},function (err) { if(!err){ console.log("修改成功~~~"); } });
或者对document对象进行修改,再执行save()方法
doc.age = 28;
doc.save();
6.3 删除
remove([callback])
doc.remove(function (err) { if(!err){ console.log("删除成功~~~"); } });
6.4 其他
get(name) - 获取文档中的指定属性值 set(name , value) - 设置文档的指定的属性值 id - 获取文档的_id属性值 toJSON() ****** - 转换为一个JSON对象 toObject() - 将Document对象转换为一个普通的JS对象【转换为普通的js对象以后,注意所有的Document对象的方法或属性都不能使用了】
console.log(doc.get("age")); console.log(doc.age); doc.set("name","猪九戒"); doc.name = "hahaha"; console.log(doc._id); var j = doc.toJSON(); console.log(j); var o = doc.toObject(); console.log(o); doc = doc.toObject(); delete doc.address; console.log(doc._id);
参考: