Mongoose基础入门

Posted 小火柴的蓝色理想

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Mongoose基础入门相关的知识,希望对你有一定的参考价值。

前面的话

  Mongoose是在node.js异步环境下对mongodb进行便捷操作的对象模型工具。本文将详细介绍如何使用Mongoose来操作MongoDB

 

NodeJS驱动

  在介绍Mongoose之前,首先介绍使用NodeJS操作MongoDB的方法

  如果使用程序操作数据库,就要使用MongoDB驱动。MongoDB驱动实际上就是为应用程序提供的一个接口,不同的语言对应不同的驱动,NodeJS驱动不能应用在其他后端语言中

  首先,安装mongodb

npm install mongodb

  接着,使用require()方法引入mongodb数据库;然后使用MongoClient对象的connect()方法连接mongodb;最后通过node来对mongodb进行异步的增删改查

  在mongodb数据库中建立db1数据库,然后通过以下代码,建立col集合,并插入{"a":1}文档

var mongodb = require(\'mongodb\');
mongodb.MongoClient.connect("mongodb://localhost/db1",function(err,db){
    if(!err){
        db.collection("col").insert({"a":1},function(err,result){
            if(!err){
                console.log(result);
            }
        })
    }
})

  最后返回结果如下

{ result: { ok: 1, n: 1 },
  ops: [ { a: 1, _id: 597077dc271d092728caa362 } ],
  insertedCount: 1,
  insertedIds: [ 597077dc271d092728caa362 ] }

 

概述

  Mongoose是NodeJS的驱动,不能作为其他语言的驱动。Mongoose有两个特点

  1、通过关系型数据库的思想来设计非关系型数据库

  2、基于mongodb驱动,简化操作

  Mongooose中,有三个比较重要的概念,分别是Schema、Model、Entity。它们的关系是:Schema生成Model,Model创造Document,Model和Document都可对数据库操作造成影响,但Model比Document更具操作性

  Schema用于定义数据库的结构。类似创建表时的数据定义(不仅仅可以定义文档的结构和属性,还可以定义文档的实例方法、静态模型方法、复合索引等),每个Schema会映射到mongodb中的一个collection,Schema不具备操作数据库的能力

  Model是由Schema编译而成的构造器,具有抽象属性和行为,可以对数据库进行增删查改。Model的每一个实例(instance)就是一个文档document

  Document是由Model创建的实体,它的操作也会影响数据库

 

安装

  安装nodejsmongodb之后 ,使用npm来安装mongoose

npm install mongoose

  安装成功后,就可以通过 require(\'mongoose\') 来使用

 

连接数据库

  使用require()方法在项目中包含mongoose后,接下来使用connect()方法连接到MongoDB数据库

【connect()】

mongoose.connect(url);

  connect()最简单的使用方式,就是只要传入url参数即可,如下所示。连接到本地localhost的db1服务器

mongoose.connect(\'mongodb://localhost/db1\');

  如果还需要传递用户名、密码,则可以使用如下方式

mongoose.connect(\'mongodb://username:password@host:port/database?options...\');

  connect()方法还接受一个选项对象options,该对象将传递给底层驱动程序。这里所包含的所有选项优先于连接字符串中传递的选项

mongoose.connect(uri, options);

  可用选项如下所示

 db            -数据库设置
 server        -服务器设置
 replset       -副本集设置
 user          -用户名
 pass          -密码
 auth          -鉴权选项
 mongos        -连接多个数据库
 promiseLibrary
var options = {
  db: { native_parser: true },
  server: { poolSize: 5 },
  replset: { rs_name: \'myReplicaSetName\' },
  user: \'myUserName\',
  pass: \'myPassword\'
}
mongoose.connect(uri, options);

  如果要连接多个数据库,只需要设置多个url以,隔开,同时设置mongos为true

mongoose.connect(\'urlA,urlB,...\', {
   mongos : true 
})

  connect()函数还接受一个回调参数

mongoose.connect(uri, options, function(error) {

});

   执行下列代码后,控制台输出“连接成功”

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://localhost/test", function(err) {
    if(err){
        console.log(\'连接失败\');
    }else{
        console.log(\'连接成功\');
    }
});

   如果开启鉴权控制,以用户名"u1",密码"123456"登录\'db1\'数据库。执行代码后,控制台输出“连接成功”

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(err){
        console.log(\'连接失败\');
    }else{
        console.log(\'连接成功\');
    }
});

【disconnect()】 

mongoose.disconnect()

   使用disconnect()方法可以断开连接

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(err){
        console.log(\'连接失败\');
    }else{
        console.log(\'连接成功\');
    }
});
setTimeout(function(){
    mongoose.disconnect(function(){
        console.log("断开连接");
    })
}, 2000);

 

Schema

  Schema主要用于定义MongoDB中集合Collection里文档document的结构  

  定义Schema非常简单,指定字段名和类型即可,支持的类型包括以下8种

String      字符串
Number      数字    
Date        日期
Buffer      二进制
Boolean     布尔值
Mixed       混合类型
ObjectId    对象ID    
Array       数组

  通过mongoose.Schema来调用Schema,然后使用new方法来创建schema对象

var mongoose = require(\'mongoose\');
var Schema = mongoose.Schema;

var mySchema = new Schema({
  title:  String,
  author: String,
  body:   String,
  comments: [{ body: String, date: Date }],
  date: { type: Date, default: Date.now },
  hidden: Boolean,
  meta: {
    votes: Number,
    favs:  Number
  }
});

  [注意]创建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\' });

【timestamps】

  在schema中设置timestamps为true,schema映射的文档document会自动添加createdAt和updatedAt这两个字段,代表创建时间和更新时间

var UserSchema = new Schema(
  {...},
  { timestamps: true }
);

【_id】

  每一个文档document都会被mongoose添加一个不重复的_id,_id的数据类型不是字符串,而是ObjectID类型。如果在查询语句中要使用_id,则需要使用findById语句,而不能使用find或findOne语句

  

Model

  模型Model是根据Schema编译出的构造器,或者称为类,通过Model可以实例化出文档对象document

  文档document的创建和检索都需要通过模型Model来处理

【model()】

mongoose.model()

  使用model()方法,将Schema编译为Model。model()方法的第一个参数是模型名称

  [注意]一定要将model()方法的第一个参数和其返回值设置为相同的值,否则会出现不可预知的结果

  Mongoose会将集合名称设置为模型名称的小写版。如果名称的最后一个字符是字母,则会变成复数;如果名称的最后一个字符是数字,则不变;如果模型名称为"MyModel",则集合名称为"mymodels";如果模型名称为"Model1",则集合名称为"model1"

var schema = new mongoose.Schema({ num:Number, name: String, size: String});
var MyModel = mongoose.model(\'MyModel\', schema);

【实例化文档document】

  通过对原型Model1使用new方法,实例化出文档document对象

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(err){
        console.log(\'连接失败\');
    }else{
        console.log(\'连接成功\');
        var schema = new mongoose.Schema({ num:Number, name: String, size: String});
        var MyModel = mongoose.model(\'MyModel\', schema);
        var doc1 = new MyModel({ size: \'small\' });
        console.log(doc1.size);//\'small\'
    }
});

【文档保存】

  通过new Model1()创建的文档doc1,必须通过save()方法,才能将创建的文档保存到数据库的集合中,集合名称为模型名称的小写复数版

  回调函数是可选项,第一个参数为err,第二个参数为保存的文档对象

save(function (err, doc) {})
var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ num:Number, name: String, size: String });
        var MyModel = mongoose.model(\'MyModel\', schema);
        var doc1 = new MyModel({ size: \'small\' });
        doc1.save(function (err,doc) {
        //{ __v: 0, size: \'small\', _id: 5970daba61162662b45a24a1 }
          console.log(doc);
        })
    }
});

  由下图所示,db1数据库中的集合名称为mymodels,里面有一个{size:"small"}的文档

 

自定义方法

【实例方法】

  Model的实例是document,内置实例方法有很多,如 save,可以通过Schema对象的methods属性给实例自定义扩展方法

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ num:Number, name: String, size: String });        
        schema.methods.findSimilarSizes = function(cb){
            return this.model(\'MyModel\').find({size:this.size},cb);
        }
        var MyModel = mongoose.model(\'MyModel\', schema);
        var doc1 = new MyModel({ name:\'doc1\', size: \'small\' });
        var doc2 = new MyModel({ name:\'doc2\', size: \'small\' });
        var doc3 = new MyModel({ name:\'doc3\', size: \'big\' });
        doc1.save();
        doc2.save();
        doc3.save();
        setTimeout(function(){
            doc1.findSimilarSizes(function(err,docs){
                docs.forEach(function(item,index,arr){
                    //doc1
                    //doc2
                     console.log(item.name)        
                })
            })  
        },0)  
    }
});

【静态方法】

  通过Schema对象的statics属性给 Model 添加静态方法

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ num:Number, name: String, size: String });        
        schema.statics.findByName = function(name,cb){
            return this.find({name: new RegExp(name,\'i\')},cb);
        }
        var MyModel = mongoose.model(\'MyModel\', schema);
        var doc1 = new MyModel({ name:\'doc1\', size: \'small\' });
        var doc2 = new MyModel({ name:\'doc2\', size: \'small\' });
        var doc3 = new MyModel({ name:\'doc3\', size: \'big\' });
        doc1.save();
        doc2.save();
        doc3.save();
        setTimeout(function(){
            MyModel.findByName(\'doc1\',function(err,docs){
                //[ { _id: 5971e68f4f4216605880dca2,name: \'doc1\',size: \'small\',__v: 0 } ]
                console.log(docs);
            })  
        },0)  
    }
});

  由上所示,实例方法和静态方法的区别在于,静态方法是通过Schema对象的statics属性model添加方法,实例方法是通过Schema对象的methods是给document添加方法

【查询方法】

  通过schema对象的query属性,给model添加查询方法

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        schema.query.byName = function(name){
            return this.find({name: new RegExp(name)});
        }
        var temp = mongoose.model(\'temp\', schema);   
        temp.find().byName(\'huo\').exec(function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 },
            // { _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 30 } ]
            console.log(docs);
        })  

    }           
});

 

文档新增

  文档新增有三种方法,一种是使用上面介绍过的文档的save()方法,另一种是使用模型model的create()方法,最后一种是模型model的insertMany()方法

【save()】

  [注意]回调函数可以省略

save([options], [options.safe], [options.validateBeforeSave], [fn])

  新建{age:10,name:\'save\'}文档,并保存

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);
        //使用链式写法    
        new temp({age:10,name:\'save\'}).save(function(err,doc){
            //[ { _id: 59720bc0d2b1125cbcd60b3f, age: 10, name: \'save\', __v: 0 } ]
            console.log(doc);        
        });         
    }           
});

【create()】

  使用save()方法,需要先实例化为文档,再使用save()方法保存文档。而create()方法,则直接在模型Model上操作,并且可以同时新增多个文档

Model.create(doc(s), [callback])

  新增{name:"xiaowang"},{name:"xiaoli"}这两个文档

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);   
        temp.create({name:"xiaowang"},{name:"xiaoli"},function(err,doc1,doc2){
            //{ __v: 0, name: \'xiaowang\', _id: 59720d83ad8a953f5cd04664 }
            console.log(doc1); 
            //{ __v: 0, name: \'xiaoli\', _id: 59720d83ad8a953f5cd04665 }
            console.log(doc2); 
        });       
    }           
});

【insertMany()】

Model.insertMany(doc(s), [options], [callback])

  新增{name:"a"},{name:"b"}这两个文档

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);   
        temp.insertMany([{name:"a"},{name:"b"}],function(err,docs){
            //[ { __v: 0, name: \'a\', _id: 59720ea1bbf5792af824b30c },
            //{ __v: 0, name: \'b\', _id: 59720ea1bbf5792af824b30d } ]
            console.log(docs); 
        });       

    }           
});

  

文档查询

  使用Mongoose来查找文档很容易,有以下3种方法可供选择

find()
findById()
findOne()

【find()】

  第一个参数表示查询条件,第二个参数用于控制返回的字段,第三个参数用于配置查询参数,第四个参数是回调函数,回调函数的形式为function(err,docs){}

Model.find(conditions, [projection], [options], [callback])

  在数据库db1的集合temps中存在如下数据

  现在,使用find()方法找出所有数据

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);
        temp.find(function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 },
            //{ _id: 5971f93be6f98ec60e3dc86d, name: \'wang\', age: 18 },
            //{ _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 30 },
            //{ _id: 5971f93be6f98ec60e3dc86f, name: \'li\', age: 12 } ]
            console.log(docs);
        })
    }
});

  找出年龄大于18的数据

        temp.find({age:{$gte:18}},function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 },
            //{ _id: 5971f93be6f98ec60e3dc86d, name: \'wang\', age: 18 },
            //{ _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 30 }]
            console.log(docs);
        })

  找出年龄大于18且名字里存在\'huo\'的数据

        temp.find({name:/huo/,age:{$gte:18}},function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 },
            //{ _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 30 }]
            console.log(docs);
        })

  找出名字里存在\'a\'的数据,且只输出\'name\'字段

  [注意]_id字段默认输出

        temp.find({name:/a/},\'name\',function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\' },
            //{ _id: 5971f93be6f98ec60e3dc86d, name: \'wang\' } ]
            console.log(docs);
        })

  如果确实不需要_id字段输出,可以进行如下设置

        temp.find({name:/a/},{name:1,_id:0},function(err,docs){
            //[ { name: \'huochai\' }, { name: \'wang\' } ]
            console.log(docs);
        })

  找出跳过前两条数据的其他所有数据

  [注意]如果使用第三个参数,前两个参数如果没有值,需要设置为null

        temp.find(null,null,{skip:2},function(err,docs){
            //[ { _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 30 },
            //{ _id: 5971f93be6f98ec60e3dc86f, name: \'li\', age: 12 } ]
            console.log(docs);
        })

【findById()】

Model.findById(id, [projection], [options], [callback])

  显示第0个元素的所有字段

        var aIDArr = [];
        temp.find(function(err,docs){
            docs.forEach(function(item,index,arr){
                aIDArr.push(item._id);
            })
            temp.findById(aIDArr[0],function(err,doc){
                //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 }
                console.log(doc);
            })            
        })

  以上代码的另一种写法如下

        var aIDArr = [];
        temp.find(function(err,docs){
            docs.forEach(function(item,index,arr){
                aIDArr.push(item._id);
            })
            temp.findById(aIDArr[0]).exec(function(err,doc){
                //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 }
                console.log(doc);
            })            
        })

  只输出name字段

            temp.findById(aIDArr[0],{name:1,_id:0},function(err,doc){
                //{  name: \'huochai\'}
                console.log(doc);
            })            

  或者写成下面这种形式

            temp.findById(aIDArr[0],{name:1,_id:0}).exec(function(err,doc){
                //{  name: \'huochai\'}
                console.log(doc);
            })            

  输出最少的字段

            temp.findById(aIDArr[0],{lean:true},function(err,doc){
                //{ _id: 5971f93be6f98ec60e3dc86c }
                console.log(doc);
            })   
            temp.findById(aIDArr[0],{lean:true}).exec(function(err,doc){
                //{ _id: 5971f93be6f98ec60e3dc86c }
                console.log(doc);
            })     

【findOne()】

  该方法返回查找到的所有实例的第一个

Model.findOne([conditions], [projection], [options], [callback])

  找出age>20的文档中的第一个文档

temp.findOne({age:{$gt : 20}},function(err,doc){
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 }
    console.log(doc);
})   
temp.findOne({age:{$gt : 20}}).exec(function(err,doc){
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 27 }
    console.log(doc);
})  

  找出age>20的文档中的第一个文档,且只输出name字段

temp.findOne({age:{$gt : 20}},{name:1,_id:0},function(err,doc){
    //{ name: \'huochai\' }
    console.log(doc);
})   
temp.findOne({age:{$gt : 20}},{name:1,_id:0}).exec(function(err,doc){
    //{ name: \'huochai\' }
    console.log(doc);
})     

  找出age>20的文档中的第一个文档,且输出包含name字段在内的最短字段

temp.findOne({age:{$gt : 20}},"name",{lean:true},function(err,doc){
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\' }
    console.log(doc);
})   
temp.findOne({age:{$gt : 20}},"name").lean().exec(function(err,doc){
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\' }
    console.log(doc);
})   

  文档查询中,常用的查询条件如下

$or    或关系
$nor    或关系取反
$gt    大于
$gte    大于等于
$lt    小于
$lte    小于等于
$ne    不等于
$in    在多个值范围内
$nin    不在多个值范围内
$all    匹配数组中多个值
$regex   正则,用于模糊查询
$size   匹配数组大小
$maxDistance 范围查询,距离(基于LBS)
$mod    取模运算
$near    邻域查询,查询附近的位置(基于LBS)
$exists   字段是否存在
$elemMatch 匹配内数组内的元素
$within   范围查询(基于LBS)
$box     范围查询,矩形范围(基于LBS)
$center   范围醒询,圆形范围(基于LBS)
$centerSphere 范围查询,球形范围(基于LBS)
$slice    查询字段集合中的元素(比如从第几个之后,第N到第M个元素

【$where】

  如果要进行更复杂的查询,需要使用$where操作符,$where操作符功能强大而且灵活,它可以使用任意的javascript作为查询的一部分,包含JavaScript表达式的字符串或者JavaScript函数

  使用字符串

temp.find({$where:"this.x == this.y"},function(err,docs){
    //[ { _id: 5972ed35e6f98ec60e3dc887,name: \'wang\',age: 18,x: 1,y: 1 },
    //{ _id: 5972ed35e6f98ec60e3dc889, name: \'li\', age: 20, x: 2, y: 2 } ]
    console.log(docs);
}) 
temp.find({$where:"obj.x == obj.y"},function(err,docs){
    //[ { _id: 5972ed35e6f98ec60e3dc887,name: \'wang\',age: 18,x: 1,y: 1 },
    //{ _id: 5972ed35e6f98ec60e3dc889, name: \'li\', age: 20, x: 2, y: 2 } ]
    console.log(docs);
}) 

  使用函数

temp.find({$where:function(){
        return obj.x !== obj.y;
    }},function(err,docs){
    //[ { _id: 5972ed35e6f98ec60e3dc886,name: \'huochai\',age: 27,x: 1,y: 2 },
    //{ _id: 5972ed35e6f98ec60e3dc888, name: \'huo\', age: 30, x: 2, y: 1 } ]
    console.log(docs);
}) 
temp.find({$where:function(){
        return this.x !== this.y;
    }},function(err,docs){
    //[ { _id: 5972ed35e6f98ec60e3dc886,name: \'huochai\',age: 27,x: 1,y: 2 },
    //{ _id: 5972ed35e6f98ec60e3dc888, name: \'huo\', age: 30, x: 2, y: 1 } ]
    console.log(docs);
}) 

 

文档更新

  文档更新可以使用以下几种方法

update()
updateMany()
find() + save() updateOne()
findOne() + save() findByIdAndUpdate() fingOneAndUpdate()

 【update()】

  第一个参数conditions为查询条件,第二个参数doc为需要修改的数据,第三个参数options为控制选项,第四个参数是回调函数

Model.update(conditions, doc, [options], [callback])

  options有如下选项

   safe (boolean): 默认为true。安全模式。
  upsert (boolean): 默认为false。如果不存在则创建新记录。
  multi (boolean): 默认为false。是否更新多个查询记录。
  runValidators: 如果值为true,执行Validation验证。
  setDefaultsOnInsert: 如果upsert选项为true,在新建时插入文档定义的默认值。
  strict (boolean): 以strict模式进行更新。
  overwrite (boolean): 默认为false。禁用update-only模式,允许覆盖记录。

  数据库temps中现有数据如下

  现在使用update()方法查询age大于20的数据,并将其年龄更改为40岁

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);   
        temp.update({age:{$gte:20}},{age:40},function(err,raw){
            //{ n: 1, nModified: 1, ok: 1 }
            console.log(raw);
        })

    }           
});

  经过以上操作,数据库结果如下。只有第一个数据更改为40岁。而第三个数据没有发生变化

  如果要同时更新多个记录,需要设置options里的multi为true。下面将名字中有\'a\'字符的年龄设置为10岁

var mongoose = require(\'mongoose\');
mongoose.connect("mongodb://u1:123456@localhost/db1", function(err) {
    if(!err){
        var schema = new mongoose.Schema({ age:Number, name: String});        
        var temp = mongoose.model(\'temp\', schema);   
        temp.update({name:/a/},{age: 10},{multi:true},function(err,raw){
            //{ n: 2, nModified: 2, ok: 1 }
            console.log(raw);
        })

    }           
});

  如果设置的查找条件,数据库里的数据并不满足,默认什么事都不发生

temp.update({age:100},{name: "hundred"},function(err,raw){
    //{ n: 0, nModified: 0, ok: 1 }
    console.log(raw);
})

  如果设置options里的upsert参数为true,若没有符合查询条件的文档,mongo将会综合第一第二个参数向集合插入一个新的文档

temp.update({age:100},{name: "hundred"},{upsert:true},function(err,raw){
    //{ n: 1, nModified: 0,upserted: [ { index: 0, _id: 5972c202d46b621fca7fc8c7 } ], ok: 1 }
    console.log(raw);
})
temp.update({name:/aa/},{age: 0},{upsert:true},function(err,raw){
    //{ n: 1, nModified: 0,upserted: [ { index: 0, _id: 5972c288d46b621fca7fdd8f } ], ok: 1 }
    console.log(raw);
})

  [注意]update()方法中的回调函数不能省略,否则数据不会被更新。如果回调函数里并没有什么有用的信息,则可以使用exec()简化代码

temp.update({name:/aa/},{age: 0},{upsert:true}).exec();

【updateMany()】

  updateMany()与update()方法唯一的区别就是默认更新多个文档,即使设置{multi:false}也无法只更新第一个文档

Model.updateMany(conditions, doc, [options], [callback])

  将数据库中名字中带有\'huo\'的数据,年龄变为50岁

temp.updateMany({name:/huo/},{age:50},function(err,raw){
    //{ n: 2, nModified: 2, ok: 1 }
    console.log(raw);
});

【find() + save()】

  如果需要更新的操作比较复杂,可以使用find()+save()方法来处理,比如找到年龄小于30岁的数据,名字后面添加\'30\'字符

temp.find({age:{$lt:20}},function(err,docs){
    //[ { _id: 5971f93be6f98ec60e3dc86d, name: \'wang\', age: 10 },
    //{ _id: 5971f93be6f98ec60e3dc86f, name: \'li\', age: 12 }]
    console.log(docs);
    docs.forEach(function(item,index,arr){
        item.name += \'30\';
        item.save();
    })
    //[ { _id: 5971f93be6f98ec60e3dc86d, name: \'wang30\', age: 10 },
    // { _id: 5971f93be6f98ec60e3dc86f, name: \'li30\', age: 12 }]
    console.log(docs);
});

【updateOne()】

   updateOne()方法只能更新找到的第一条数据,即使设置{multi:true}也无法同时更新多个文档

  将数据库中名字中带有\'huo\'的数据,年龄变为60岁

temp.updateOne({name:/huo/},{age:60},function(err,raw){
    //{ n: 1, nModified: 1, ok: 1 }
    console.log(raw);
});

【findOne() + save()】

  如果需要更新的操作比较复杂,可以使用findOne()+save()方法来处理,比如找到名字为\'huochai\'的数据,年龄加100岁

temp.findOne({name:\'huochai\'},function(err,doc){
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 10 }
    console.log(doc);
    doc.age += 100;
    doc.save();
    //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 110 }
    console.log(doc);
});

【findOneAndUpdate()】

  fineOneAndUpdate()方法的第四个参数回调函数的形式如下function(err,doc){}

Model.findOneAndUpdate([conditions], [update], [options], [callback])

【findByIdAndUpdate】

   fineByIdAndUpdate()方法的第四个参数回调函数的形式如下function(err,doc){}

Model.findOneAndUpdate([conditions], [update], [options], [callback])

 

文档删除

  有三种方法用于文档删除

remove()
findOneAndRemove()
findByIdAndRemove()

【remove()】

  remove有两种形式,一种是文档的remove()方法,一种是Model的remove()方法

  下面介绍Model的remove()方法,该方法的第一个参数conditions为查询条件,第二个参数回调函数的形式如下function(err){}  

model.remove(conditions, [callback])

  删除数据库中名称包括\'30\'的数据

temp.remove({name:/30/},function(err){})

  [注意]remove()方法中的回调函数不能省略,否则数据不会被删除。当然,可以使用exec()方法来简写代码

temp.remove({name:/30/}).exec()

  下面介绍文档的remove()方法,该方法的参数回调函数的形式如下function(err,doc){}

document.remove([callback])

  删除数据库中名称包含\'huo\'的数据

  [注意]文档的remove()方法的回调函数参数可以省略

temp.find({name:/huo/},function(err,doc){
    doc.forEach(function(item,index,arr){
        item.remove(function(err,doc){
            //{ _id: 5971f93be6f98ec60e3dc86c, name: \'huochai\', age: 30 }
            //{ _id: 5971f93be6f98ec60e3dc86e, name: \'huo\', age: 60 }
            console.log(doc);
        })
    })
})  

【findOneAndRemove()】

  model的remove()会删除符合条件的所有数据,如果只删除符合条件的第一条数据,则可以使用model的findOneAndRemove()方法

Model.findOneAndRemove(conditions, [options], [callback])

   集合temps现有数据如下

  现在删除第一个年龄小于20的数据

temp.findOneAndRemove({age:{$lt:20}},function(err,doc){
    //{ _id: 5972d3f3e6f98ec60e3dc873, name: \'wang\', age: 18 }
    console.log(doc);
})

  与model的remove()方法相同,回调函数不能省略,否则数据不会被删除。当然,可以使用exec()方法来简写代码

temp.findOneAndRemove({age:{$lt:20}}).exec()

【findByIdAndRemove()】

Model.findByIdAndRemove(id, [options], [callback])

  删除第0个元素

var aIDArr = [];
temp.find(function(err,docs){
    docs.forEach(function(item,index,arr){
        aIDArr.push(item._id);
    })
    temp.findByIdAndRemove(aIDArr[0],function(err,doc){
        //{ _id: 5972d754e6f98ec60e3dc882, name: \'huochai\', age: 27 }
        console.log(doc);
    })            
})

  类似的,该方法也不能省略回调函数,否则数据不会被删除。当然,可以使用exec()方法来简写代码

var aIDArr = [];
temp.find(function(err,docs){
    docs.forEach(function(item,index,arr){
        aIDArr.push(item._id);
    })
    temp.findByIdAndRemove(aIDArr[0]).exec()            
})

 

前后钩子

  前后钩子即pre()和post()方法,又称为中间件,是在执行某些操作时可以执行的函数。中间件在schema上指定,类似于静态方法或实例方法等

  可以在数据库执行下列操作时,设置前后钩子

    init
    validate
    save
    remove
    count
    find
    findOne
    findOneAndRemove
    findOneAndUpdate
    insertMany
    update

【pre()】

  以find()方法为例,在执行find()方法之前,执行pre()方法

var schema = new mongoose.Schema({ age:Number, name: String,x:Number,y:Number});  
schema.pre(\'find\',function(next){
    console.log(\'我是pre方法1\');
    next();
});
schema.pre(\'find\',function(next){
    console.log(\'我是pre方法2\');
    next();
});  
var temp = mongoose.model(\'temp\', schema);
temp.find(function(err,docs){
    console.log(docs[0]);
})    
/*
我是pre方法1
我是pre方法2
{ _id: 5972ed35e6f98ec60e3dc886,name: \'huochai\',age: 27,x: 1,y: 2 }
*/

【post()】

  post()方法并不是在执行某些操作后再去执行的方法,而在执行某些操作前最后执行的方法,post()方法里不可以使用next()

var schema = new mongoose.Schema({ age:Number, name: String,x:Number,y:Number});  
schema.post(\'find\',function(docs){
    console.log(\'我是post方法1\');
});
schema.post(\'find\',function(docs){
    console.log(\'我是post方法2\');
});
var temp = mongoose.model(\'temp\', schema);
temp.find(function(err,docs){
    console.log(docs[0]);
}) 
/*
我是post方法1
我是post方法2
{ _id: 5972ed35e6f98ec60e3dc886,name: \'huochai\',age: 27,x: 1,y: 2 }
 */   

 

查询后处理

  常用的查询后处理的方法如下所示

sort     排序
skip 跳过 limit 限制 select 显示字段 exect 执行
count 计数
distinct 去重
var schema = new mongoose.Schema({ age:Number, name: String,x:Number,y:Number});  
var temp = mongoose.model(\'temp\', schema);
temp.find(function(err,docs){
    //[ { _id: 5972ed35e6f98ec60e3dc886,name: \'huochai\',age: 27,x: 1,y: 2 },
    //{ _id: 5972ed35e6f98ec60e3dc887,name: \'wang\',age: 18,x: 1,y: 1 },
    //{ _id: 5972ed35e6f98ec60e3dc888, name: \'huo\', age: 30, x: 2, y: 1 },
    //{ _id: 5972ed35e6f98ec60e3dc889, name: \'li\', age: 20, x: 2

以上是关于Mongoose基础入门的主要内容,如果未能解决你的问题,请参考以下文章

Express实战 - 应用案例- realworld-API - 路由设计 - mongoose - 数据验证 - 密码加密 - 登录接口 - 身份认证 - token - 增删改查API(代码片段

react_新手入门教程05——react + express + mongoose 实现CURD

JSP基础--JSP入门

Java基础入门五)之方法以及递归算法

推荐net开发cad入门阅读代码片段

mongoose入门以及 mongoose 实现数据 的增删改查