zoukankan      html  css  js  c++  java
  • Mongoose使用——nodejs结合mongodb

     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);

    参考:

    官方文档

    Mongoose基础入门

    尚硅谷培训课程

  • 相关阅读:
    Animate.css 一款强大的预设css3动画库
    关于js返回上一页的实现方法
    jquery判断字符串中是否包含特定字符的方法总结
    去掉select在苹果手机上的原生样式
    html5中如何去掉input type date默认样式
    JS和jQuery中ul li遍历获取对应的下角标
    滚动一定的高度底色递增
    喵哈哈村的狼人杀大战(5)
    喵哈哈村的狼人杀大战(2)
    One Card Poker
  • 原文地址:https://www.cnblogs.com/zjfjava/p/8733613.html
Copyright © 2011-2022 走看看