返回顶部
首页 > 资讯 > 前端开发 > node.js >Mongoose学习全面理解(推荐)
  • 600
分享到

Mongoose学习全面理解(推荐)

Mongoose 2022-06-04 17:06:16 600人浏览 薄情痞子
摘要

一、创建schemas 创建schemas的方式: var userSchema = new monGoose.Schema({ name: String, email: String,

一、创建schemas

创建schemas的方式:


 var userSchema = new monGoose.Schema({
   name: String,
   email: String,
   createdOn: Date
 });

schemas中的数据类型有以下几种:

String Number Date Boolean Buffer ObjectId Mixed Array

特别需要说明一下ObjectId类型和Mixed类型以及Array类型,在schemas中声明这几种类型的方式如下:


//ObjectId就类似于唯一键值
projectSchema.add({
  owner: mongoose.Schema.Types.ObjectId
});
//混合类型,顾名思义,就是说里面可以放置任意类型的数据,有两种方式创建该类型数据
//方式一:直接赋予一个空的字面量对象
vardjschema= new mongoose.Schema({
  mixedUp: {}
});
//方式二:根据Schemas.Types中值来赋予
vardjSchema= new mongoose.Schema({
  mixedUp: Schema.Types.Mixed
});
//Array类型数据有两种创建方式,一种是简单数组创建:
var userSchema = new mongoose.Schema({
  name: String,
  emailAddresses: [String]
});
//第二种方式就是复杂类型数据数组,例如我们可以再数组中添加不同类型的schemas:
var emailSchema = new mongoose.Schema({
  email: String,
  verified: Boolean
});
var userSchema = new mongoose.Schema({
  name: String,
  emailAddresses: [emailSchema]
});
//注意:如果定义一个空的数据的话,则会创建为一个混合类型数据的数组:
var emailSchema = new mongoose.Schema({
  email: String,
  verified: Boolean
});
var userSchema = new mongoose.Schema({
  name: String,
  emailAddresses: [emailSchema]
});

我们可以给schema创建静态方法,这个静态方法将来会用在Model中,创建该静态方法需要在创建完成schema之后,在Model编译之前:


 projectSchema.statics.findByUserID = function (userid, callback) {
  this.find({ createdBy: userid }, '_id projectName', {sort: 'modifiedOn'}, callback);
 };

在其对应的模型创建完成并编译后,我们就可以像下面这样来调用该静态方法了:


Model.findByUserID(userid,callback);

该静态方法会返回一个JSON格式的数据,这在我们使用ajax技术来加载网页数据的时候会比较方便,就像下面这样:


//路由规则:app.get('/project/byuser/:userid', project.byUser);
exports.byUser = function (req, res) {
  console.log("Getting user projects");
  if (req.params.userid){
    Project.findByUserID(req.params.userid,function (err, projects) {
      if(!err){
        console.log(projects);
        res.json(projects);
      }else{
        console.log(err);
        res.json({"status":"error", "error":"Error finding projects"});
      }
    });
  }else{
    console.log("No user id supplied");
    res.json({"status":"error", "error":"No user id supplied"});
  }
};

二、创建Model

创建Model很简单:


Mongoose.Model('User', userSchema);

参数一为Model的名字,参数二为生成Model所需要的schema,Model就像是schema所编译而成的一样。

mongoose连接数据库是有两种方式的:


//方式一:
var dbURI = 'mongoDB://localhost/mydatabase';
mongoose.connect(dbURI);
//方式二:
var dbURI = 'mongodb://localhost/myadmindatabase';
var adminConnection = mongoose.createConnection(dbURI);
//如果需要声明端口号:
var dbURI = 'mongodb://localhost:27018/mydatabase';
//如果需要定义用户名和密码:
var dbURI = 'mongodb://username:passWord@localhost/mydatabase';
//也可以像下面这样传一个对象类型的参数:
var dbURI = 'mongodb://localhost/mydatabase';
var dbOptions = {'user':'db_username','pass':'db_password'};
mongoose.connect(dbURI, dbOptions);

根据连接数据库的方式,我们可以得到第二种创建Model的方式,就是使用数据库连接的引用名来创建:


adminConnection.model( 'User', userSchema );

默认情况下mongoose会根据我们传入的Model名字来生成collection名字,在上面的代码中就会生成名为users(全为小写字母)的collection(集合);

有两种方法能让我们自定义collection的名字。


//方式一,在创建schema的时候定义collection的名字:
var userSchema = new mongoose.Schema({
  name: String,
  email: {type: String, unique:true}
},
{
  collection: 'myuserlist'
});
//方式二,在创建Model的时候定义collection的名字:
mongoose.model( 'User', userSchema, 'myuserlist' );

创建Model实例:


var user = new User({ name: 'Simon' });

user就是模型User的一个实例,它具有mongoose中模型所具有的一些方法,例如保存实例:


 user.save(function (err) {
   if (err) return handleError(err);
 });

模型也具有一些常用的增删查改的方法:


User.findOne({'name' : 'Sally', function(err,user) {
  if(!err){
    console.log(user);
  }
});
User.find({}, function(err, users) {
  if(!err){
    console.log(users);
  }
});

可以使用链式方式使用这些方法,例如:


var newUser = new User({
  name: 'Simon Holmes',
  email: 'simon@theholmesoffice.com',
  lastLogin : Date.now()
}).save( function( err ){
  if(!err){
    console.log('User saved!');
  }
});

上面的代码创建了一个模型实例,然后进行保存。我们有一个更为简介的方式来完成这项工作,就是使用Model.create()方法:


User.create({
  name: 'Simon Holmes',
  email: 'simon@theholmesoffice.com',
  lastLogin : Date.now()
}, function( err, user ){
  if(!err){
    console.log('User saved!');
    console.log('Saved user name: ' + user.name);
    console.log('_id of saved user: ' + user._id);
  }
});

三、查找数据和读取数据的方法

1.使用QueryBuilder接口来查找数据

先看看下面的代码:


var myQuery = User.find({'name' : 'Simon Holmes'});
myQuery.where('age').gt(18);
myQuery.sort('-lastLogin');
myQuery.select('_id name email');
myQuery.exec(function (err, users){
  if (!err){
    console.log(users); // output array of users found
  }
});

代码中,我们查找名字为"Simon Holmes",并且年龄大于18岁,查找结果根据lastLogin降序排列,只获取其中的_id, name, email三个字段的值,上面的代码只有在调用exec方法后才真正执行数据库的查询。

当然我们可以使用链式的方式来改写上面的代码,代码会更加简洁:


User.find({'name' : 'Simon Holmes'})
.where('age').gt(18)
.sort('-lastLogin')
.select('_id name email')
.exec(function (err, users){
  if (!err){
    console.log(users); // output array of users found
  }
});

上面代码中的第一行创建了一个queryBuilder.通过使用这个queryBuilder,我们就可以执行一些比较复杂的查找工作,在创建完成这个queryBuilder之后,查询操作并没有马上执行,而是待到执行exec方法时才会去执行数据库的查找。

当然也有另外一种方式能够直接查找数据库的,就是直接在查找方法中添加回调函数,使用方式为:


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

下面举一个简单例子:


 User.find({'name', 'simon holmes'}, function(err, user) {});

另一个稍微复杂的例子:


 User.find({'name', 'simon holmes'}, 'name email',function(err, user) {
   //console.log('some thing');
 });

另一个更加复杂的例子,包含查询结果的排序


User.find({'name' : 'Simon Holmes'},
  null, // 如果使用null,则会返回所有的字段值
  {sort : {lastLogin : -1}}, // 降序排序
  function (err, users){
    if (!err){console.log(users);}
  });

列举几个比较实用的查找方法:


 Model.find(query);
 Model.findOne(query);//返回查找到的所有实例的第一个
 Model.findById(ObjectID);//根据ObjectId查找到唯一实例

例如:


 User.findOne({'email' : req.body.Email},
 '_id name email',
 function(err, user) {
   //todo
 });

2.更新数据

有三种方式来更新数据:

(1)update(conditions,update,options,callback);

该方法会匹配到所查找的内容进行更新,不会返回数据;

(2)findOneAndUpdate(conditions,update,options,callback);

该方法会根据查找去更新数据库,另外也会返回查找到的并未改变的数据;

(3)findByIdAndUpdate(conditions,update,options,callback);

该方法跟上面的findOneAndUpdate方法功能一样,不过他是根据ID来查找文档并更新的。

三个方法都包含四个参数,一下稍微说明一下几个参数的意思:

conditions:查询条件 update:更新的数据对象,是一个包含键值对的对象 options:是一个声明操作类型的选项,这个参数在下面再详细介绍 callback:回调函数

对于options参数,在update方法中和findOneAndUpdate、findByIdAndUpdate两个方法中的可选设置是不同的;


//在update方法中,options的可选设置为:
{
safe:true|false, //声明是否返回错误信息,默认true
upsert:false|true, //声明如果查询不到需要更新的数据项,是否需要新插入一条记录,默认false
multi:false|true, //声明是否可以同时更新多条记录,默认false
strict:true|false //声明更新的数据中是否可以包含在schema定义之外的字段数据,默认true
}
//对于findOneAndUpdate、findByIdAndUpdate这两个方法,他们的options可选设置项为:
{
new:true|false, //声明返回的数据时更新后的该是更新前的,如果为true则返回更新后的,默认true
upsert:false|trure, 
sort:javascriptObject, //如果查询返回多个文档记录,则可以进行排序,在这里是根据传入的javascript object对象进行排序
select:String //这里声明要返回的字段,值是一个字符串
}

下面举个例子:


 User.update({_id:user._id},{$set: {lastLogin: Date.now()}},function(){});

3.数据删除

跟更新数据一样,也有三种方法给我们删除数据:


remove();
findOneAndRemove();
findByIdAndRemove();

remove方法有两种使用方式,一种是用在模型上,另一种是用在模型实例上,例如:


User.remove({ name : /Simon/ } , function (err){
  if (!err){
    // 删除名字中包含simon的所有用户
  }
});

User.findOne({ email : 'simon@theholmesoffice.com'},function (err,user){
  if (!err){
    user.remove( function(err){
      // 删除匹配到该邮箱的第一个用户
    });
  }
});

接下来看一下findOneAndRemove方法:


User.findOneAndRemove({name : /Simon/},{sort : 'lastLogin', select : 'name email'},function (err, user){
  if (!err) {
    console.log(user.name + " removed");
    // Simon Holmes removed
  };
});

另外一个findByIdAndRemove方法则是如出一辙的。


User.findByIdAndRemove(req.body._id,function (err, user) {
  if(err){
    console.log(err);
    return;
  }
  console.log("User deleted:", user);
});

四、数据验证

1.mongoose内置数据验证

在mongoose中,数据验证这一层是放在schema中的,mongoose已经帮我们做了很多内置的数据验证,有一些验证是针对某些数据类型的,也有一些是针对所有数据类型的。

能够作用在所有数据类型上的验证有require,意思就是该字段是否是必须的,例如:


email: { type: String, unique: true, required: true }

上面的代码就定义了一个email是必须的schema.

下面再分别介绍一下mongoose内置的一些数据验证类型。

数字类型schemasType,对于Number类型的数据,具有min,max提供用来界定最大最小值:


 var teenSchema = new Schema({
   age : {type: Number, min: 13, max:19}
 });

字符串类型SchemasType,对于该类型数据,mongoose提供了两种验证器:

match:可使用正则表达式来匹配字符串是否符合该正则表达式的规则 enum:枚举出字符串可使用的一些值

分别举例如下:


var weekdaySchema = new Schema({
  day : {type: String, match: /^(mon|tues|wednes|thurs|fri)day$/i}
});

var weekdays = ['monday', 'tuesday', 'wednesday', 'thursday','friday'];
var weekdaySchema = new Schema({
  day : {type: String, enum: weekdays}
});

在我们进行一些数据库的时候,如果有错误,可能会返回一些错误信息,这些信息封装在一个对象中,该对象的数据格式大致如下:


{ 
  message: 'Validation failed',
  name: 'ValidationError',
  errors:{ 
    email:{
      message: 'Validator "required" failed for path email',
      name: 'ValidatorError',
      path: 'email',
      type: 'required' 
    },
    name:{ 
      message: 'Validator "required" failed for path name',
      name: 'ValidatorError',
      path: 'name',
      type: 'required' 
    } 
  } 
}

知道该错误信息的具体格式之后,我们可以从中得出我们想要的信息并反馈到控制台。


if(err){
  Object.keys(err.errors).forEach(function(key) {
    var message = err.errors[key].message;
    console.log('Validation error for "%s": %s', key, message);
  });
}

2.自定义数据验证

最简单的自定义数据验证方式就是定义一个数据验证的函数,并将它传递给schema;


var lengthValidator = function(val) {
  if (val && val.length >= 5){
    return true;
  }
  return false;
};
//usage:
name: {type: String, required: true, validate: lengthValidator }

可以看到,我们只需要在schema中添加validate键值对即可,validate对应的值便是我们自定义的验证方法;

但是该形式的数据验证无法给我们提供完整的错误信息,比如errors信息中返回的type值就会成为undefined;

在此基础上如果希望错误信息中能返回一个错误描述,那我们可以稍微进行一点修改:


//code 1
validate: { validator: lengthValidator, msg: 'Too short' }

//code 2
var weekdaySchema = new Schema({
  day : {type: String, validate: {validator:/^(mon|tues|wednes|thurs|fri)day$/i, msg: 'Not a day' }
});

将validate的值修改为一个对象,并且该对象包含验证器和错误描述。

我们也可以使用另一种方式在写这些验证器,就是将验证器卸载schema外部,例如:


 var validateLength = [lengthValidator, 'Too short' ];
 var validateDay = [/^(mon|tues|wednes|thurs|fri)day$/i, 'Not a day' ];
 //usage:
 name: {type: String, required: true, validate: validateLength }
 day : {type: String, validate: validateDay }

眼睛放大,一看再看,确实没错,在validate中我们传入的是一个数组了,而不是原来的对象了。

其实就validateLength这个东东来说,他就是一个简写来的,你也可以改成下面这样:


 var validateLength = [
   {validator: lengthValidator, msg: 'Too short'}
 ];

恩,到这里,应该能明白了,将对象改为数组之后,我们便可以传递多个验证器给我们的schema了,的确如此。


 var validateUsername = [
   {validator: lengthValidator, msg: 'Too short'} ,
   {validator: /^[a-z]+$/i, msg: 'Letters only'}
 ];

我们还有另外一种方法给我们的schema提供验证器:


userSchema.path('name').validate(lengthValidator, 'Too short');
userSchema.path('name').validate(/^[a-z]+$/i, 'Letters only');

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程网。

--结束END--

本文标题: Mongoose学习全面理解(推荐)

本文链接: https://lsjlt.com/news/12826.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • Mongoose学习全面理解(推荐)
    一、创建schemas 创建schemas的方式: var userSchema = new mongoose.Schema({ name: String, email: String, ...
    99+
    2022-06-04
    Mongoose
  • 学习C++的书籍推荐
    这篇文章将为大家详细讲解有关学习C++的推荐书籍,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。Effective C++和More Effective C++, by Scott Meyers:这两本书也...
    99+
    2023-05-31
    c++ c+
  • PHP学习笔记:推荐系统与个性化推荐
    引言:在当今互联网时代,推荐系统已成为许多网站和应用程序的重要功能之一。通过运用机器学习和数据挖掘技术,推荐系统能够根据用户的行为和兴趣,将最相关的内容和产品推荐给用户,提升用户体验和网站的交互性。而个性化推荐则是推荐系统的一种重要算法,能...
    99+
    2023-10-21
    推荐系统 PHP 个性化推荐
  • 推荐两个MySQL学习资源
    高鹏(八怪)从2008年至今一直从事Oracle/MySQL相关工作,现任易极付高级DBA,Oracle OCM。有丰富的数据库性能调优,故障诊断经验,并且对MySQL源码有一定的研究。 这个系列专栏主要包含GTID部分、Event部...
    99+
    2017-10-04
    推荐两个MySQL学习资源
  • 学习linux常用命令(推荐)
    一、文件系统的管理 tips:输入命令的时候要常用tab键来补全 ls 查看目录信息 ( ls / ) ls -l 等价于 ll pwd 查看当前所处的路径 cd 切换目录 (cd /) ,如果不...
    99+
    2022-06-04
    常用命令 linux
  • Java学习之路(书籍推荐)
     前言:Java开发工程师一般负责后端开发,当然也有专门做Java Web的工程师,但是随着前后端的分离,越来越多的Java工程师需要往大后端方向发展。今天我们就来介绍一下Java后端开发者的书单。首先要感谢一下某大佬力推的后端书...
    99+
    2023-06-02
  • 墙裂推荐一波mysql学习资源
    在日常工作与学习中,无论是开发、运维、测试,还是架构师,数据库是一门必不可少的”必修课”, 也是必备的涨薪神器。在互联网公司中,开源数据库用得比较多的当属 MySQL 了。 但mysql知识面非常广,知识点也很零散,学习过程是比较曲折的,...
    99+
    2019-05-22
    墙裂推荐一波mysql学习资源
  • Go语言培训:学习资源推荐
    在当今信息化时代,计算机语言的重要性不言而喻。随着互联网和人工智能等领域的飞速发展,编程语言也变得愈发重要。而在诸多编程语言中,Go语言作为一种开发效率高、性能优越的编程语言,备受程序...
    99+
    2024-02-27
    资源推荐 go语言学习 培训建议 go语言 在线课程
  • PHP自动加载学习资源推荐
    PHP自动加载是将类、接口和特性映射到文件路径的机制。这允许您在类名或接口名之前使用use而不必手动包含文件。这使得您的代码更加简洁和易于维护。 如果你想学习PHP自动加载,这里有一些资源可以帮助你入门: 官方文档:PHP官方文档提供...
    99+
    2024-02-08
    PHP 自动加载 学习资源 入门 进阶
  • Spring-全面详解(学习总结)
    目录一、Spring介绍简介特点、主要特点为什么要学?二、IOC(依赖注入)Spring 容器IOC 和 bean介绍控制反转:容器概述:bean介绍AOP总结一、Spring介绍 ...
    99+
    2024-04-02
  • Spring全面详解(学习总结)
    Spring FrameWork一、 前言二、IOC(控制反转)2.1 对于IOC的理解2.2如何使用IOC2.3配置文件的解读2.4IOC容器创建bean的两种方式2.5从IOC容器中取bean2.6bean的属性如...
    99+
    2023-08-16
    spring java IOC AOP
  • Discuz! X1.5插件制作学习之旅 推荐
    单页制作教程 FOR Discuz! X1.5在Discuz! X1.5根目录下新建一个文件,host.php,源码内容如下 <php require './source/class/class_cor...
    99+
    2022-06-12
    Discuz 插件
  • 实时数据处理神器!Python API 学习笔记推荐
    在当今的信息时代,数据处理已经成为了各个行业的重要工作之一。而随着大数据技术的逐渐成熟,实时数据处理已经成为了越来越多企业和个人的需求。而 Python 作为一门高效的编程语言,其强大的数据处理能力更是被越来越多人所认可。本文将为大家介绍...
    99+
    2023-11-08
    api 学习笔记 实时
  • Linux运维学习路线,实用LINUX教程推荐学习收藏
    Linux操作系统在短短的几年之内得到了非常迅猛的发展,这与linux具有的良好特性是分不开的。Linux包含了UNIX的全部功能和特性。简单来说,linux具有以下主要特性:遵循GNU GPL,开放性,多任务,多用户,设备独立性,提供了丰...
    99+
    2023-06-05
  • Go语言学习路线图:系统学习方法和资源推荐
    Go语言是由谷歌公司开发的一种开源编程语言,它是一种跨平台的语言,具有高效的编译器和强大的并发特性。随着Go语言在云计算、大数据、网络编程等领域的广泛应用,越来越多的人开始学习Go语言...
    99+
    2024-02-27
    路线图 学习方法 资源推荐 go语言 网络编程 在线课程
  • 网络安全ctf比赛/学习资源整理,解题工具、比赛时间、解题思路、实战靶场、学习路线,推荐收藏!...
       对于想学习或者参加CTF比赛的朋友来说,CTF工具、练习靶场必不可少,今天给大家分享自己收藏的CTF资源,希望能对各位有所帮助。 CTF在线工具 首先给大家推荐我自己常用的3个CTF在线工具网站,内容齐全,收藏备用。 1、...
    99+
    2023-08-31
    web安全 学习 php 安全 开发语言
  • 推荐阅读《Tensorflow:实战Google深度学习框架》
    推荐阅读《Tensorflow:实战Google深度学习框架》TensorFlow是谷歌2015年开源的主流深度学习框架,目前已在谷歌、优步(Uber)、京东、小米等科技公司广泛应用。《Tensorflow实战》为使用TensorFlow深...
    99+
    2023-06-02
  • ASP 数据类型有哪些?学习教程推荐!
    ASP(Active Server Pages)是一种基于服务器端脚本的技术,用于动态生成网页。在 ASP 中,我们需要使用各种数据类型来存储和处理数据。本文将介绍 ASP 中常用的数据类型,以及相关的学习教程推荐。 一、数据类型 字符...
    99+
    2023-07-07
    数据类型 教程 unix
  • Go语言学习的五个精选课程推荐
    Go语言作为一门现代化的高级编程语言,已经在软件开发领域逐渐崭露头角。作为一个新手,学习Go语言可能会遇到一些困难,因为它具有自己独特的语法和编程概念。为了帮助新手更好地掌握Go语言,我在这里精选了五个深入学...
    99+
    2024-01-30
    Go语言入门 Go语言进阶 Go语言开发实践 网络编程
  • 肇庆学习Go语言的优质资源推荐
    肇庆学习Go语言的优质资源推荐 近年来,Go语言作为一种新兴的编程语言逐渐受到人们的关注和喜爱。作为一门强大而简洁的编程语言,它不仅在性能上表现优异,而且具有良好的并发能力和易用性,因...
    99+
    2024-02-27
    学习指南 go语言资源 教材推荐 go语言 标准库
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作