0

0

Mongoose多数据库连接与模型使用指南

DDD

DDD

发布时间:2025-07-30 14:30:01

|

991人浏览过

|

来源于php中文网

原创

Mongoose多数据库连接与模型使用指南

本教程旨在解决Mongoose在使用mongoose.createConnection建立多数据库连接时,模型实例化遇到的常见错误。我们将详细讲解如何在特定连接上正确定义和注册模型,并演示如何通过该连接实例来创建和操作模型实例,确保数据能准确地保存到指定的数据库中,避免TypeError: conn.Price is not a constructor等问题,提升多数据库应用开发的效率与稳定性。

在复杂的应用场景中,我们可能需要连接并操作多个mongodb数据库。mongoose提供了两种主要的方式来建立数据库连接:mongoose.connect() 和 mongoose.createconnection()。前者通常用于建立应用程序的默认连接,而后者则允许你创建独立的、非默认的连接实例,这对于操作多个数据库或为不同模块使用不同数据库连接的场景至关重要。

理解Mongoose连接与模型注册

当使用 mongoose.createConnection() 方法时,它会返回一个全新的 Connection 实例。这个实例独立于全局的 mongoose 对象。这意味着,通过这个连接实例注册的模型,将仅与这个特定的连接绑定,而不是全局的 mongoose 对象。

错误根源分析:

在原始问题中,用户尝试通过 conn.model('Price', priceSchema); 将 Price 模型注册到 conn 这个连接实例上。这是正确的做法。然而,在尝试实例化模型时,却使用了 new conn.Price();。这里的错误在于,conn.Price 并不是 Mongoose 提供的一种直接访问已注册模型构造函数的方式。conn 对象本身不会动态地将已注册的模型作为其属性暴露出来。

要正确地获取并使用通过 conn.model() 注册的模型构造函数,你需要:

SmartB2B行业电子商务
SmartB2B行业电子商务

SmartB2B 是一款基于PHP、MySQL、Smarty的B2B行业电子商务网站管理系统,系统提供了供求模型、企业模型、产品模型、人才招聘模型、资讯模型等模块,适用于想在行业里取得领先地位的企业快速假设B2B网站,可以运行于Linux与Windows等多重服务器环境,安装方便,使用灵活。 系统使用当前流行的PHP语言开发,以MySQL为数据库,采用B/S架构,MVC模式开发。融入了模型化、模板

下载
  1. 在注册模型时,将 conn.model('ModelName', Schema) 的返回值赋值给一个变量。这个返回值就是该模型的构造函数。
  2. 或者,在模型已经注册到特定连接后,通过 conn.model('ModelName') 再次调用来获取其构造函数。

正确的Mongoose多数据库连接与模型使用示例

下面是一个详细的示例,展示了如何使用 mongoose.createConnection() 建立多个数据库连接,并正确地定义、注册和实例化模型:

const mongoose = require('mongoose');

// 数据库连接选项
// 注意:useCreateIndex, useFindAndModify 在 Mongoose 6.0+ 中已弃用,
// useNewUrlParser, useUnifiedTopology 也是默认行为,通常不再需要显式设置。
const connectionOptions = {
    useNewUrlParser: true, // 保持兼容性,尽管在新版本中已是默认
    useUnifiedTopology: true // 保持兼容性,尽管在新版本中已是默认
};

// --- 连接到第一个数据库:db_en ---
const dbEnConn = mongoose.createConnection("mongodb://localhost/db_en", connectionOptions);

// 监听连接事件,确保连接成功或处理错误
dbEnConn.on('connected', () => {
    console.log('成功连接到数据库:db_en');
});
dbEnConn.on('error', (err) => {
    console.error('db_en 数据库连接错误:', err);
});
dbEnConn.on('disconnected', () => {
    console.log('db_en 数据库连接已断开');
});

// 定义 Price 模型的 Schema
const priceSchema = new mongoose.Schema({
    fixed: {
        1: { type: Number, default: 199 },
        3: { type: Number, default: 499 },
        6: { type: Number, default: 729 },
        12: { type: Number, default: 999 }
    }
}, { timestamps: true }); // 添加时间戳,便于追踪

// 将 Price 模型注册到 dbEnConn 连接实例上,并获取其构造函数
const PriceModel = dbEnConn.model('Price', priceSchema);

// 实例化 Price 模型并保存数据
async function savePriceData() {
    try {
        const newPriceEntry = new PriceModel({
            fixed: {
                1: 200,
                3: 500
            }
        });
        const savedPrice = await newPriceEntry.save();
        console.log('价格数据已成功保存到 db_en:', savedPrice);
    } catch (error) {
        console.error('保存价格数据时发生错误:', error);
    }
}

// --- 连接到第二个数据库:db_another ---
const dbAnotherConn = mongoose.createConnection("mongodb://localhost/db_another", connectionOptions);

// 监听连接事件
dbAnotherConn.on('connected', () => {
    console.log('成功连接到数据库:db_another');
});
dbAnotherConn.on('error', (err) => {
    console.error('db_another 数据库连接错误:', err);
});
dbAnotherConn.on('disconnected', () => {
    console.log('db_another 数据库连接已断开');
});

// 定义 User 模型的 Schema
const userSchema = new mongoose.Schema({
    name: { type: String, required: true },
    email: { type: String, required: true, unique: true }
}, { timestamps: true });

// 将 User 模型注册到 dbAnotherConn 连接实例上,并获取其构造函数
const UserModel = dbAnotherConn.model('User', userSchema);

// 实例化 User 模型并保存数据
async function saveUserData() {
    try {
        const newUserEntry = new UserModel({
            name: 'Alice',
            email: 'alice@example.com'
        });
        const savedUser = await newUserEntry.save();
        console.log('用户数据已成功保存到 db_another:', savedUser);

        // 尝试保存第二个用户,演示错误处理(如 email unique 约束)
        const anotherUser = new UserModel({
            name: 'Bob',
            email: 'alice@example.com' // 会导致唯一性约束错误
        });
        await anotherUser.save();
    } catch (error) {
        // Mongoose 错误通常有 name 和 code 属性
        if (error.code === 11000) {
            console.error('保存用户数据时发生错误: 邮箱已存在。', error.message);
        } else {
            console.error('保存用户数据时发生错误:', error);
        }
    }
}

// 运行示例
async function runExamples() {
    await savePriceData();
    await saveUserData();

    // 在实际应用中,你可能需要等待所有操作完成后再关闭连接
    // setTimeout(() => {
    //     dbEnConn.close(() => console.log('db_en 连接已关闭'));
    //     dbAnotherConn.close(() => console.log('db_another 连接已关闭'));
    // }, 5000); // 示例:5秒后关闭
}

runExamples();

注意事项与最佳实践

  1. 模型与连接的绑定: 使用 connection.model('ModelName', schema) 注册模型时,返回的就是该模型的构造函数。务必将其赋值给一个变量,后续通过这个变量来实例化模型,例如 const MyModel = myConnection.model('MyModel', mySchema); new MyModel();。
  2. 避免全局污染: 除非你有意为之,否则不要将通过 createConnection 创建的连接上的模型注册到全局 mongoose.model 中。那样会导致模型与默认连接绑定,而不是你期望的特定连接。
  3. 连接生命周期管理: 在应用程序启动时建立连接,并在应用程序关闭时优雅地关闭它们,以释放资源。监听 connected, error, disconnected 等事件有助于更好地管理连接状态。
  4. 错误处理: 对数据库操作(如 save(), find(), update() 等)进行适当的错误处理,使用 try...catch 块或 .then().catch() 链来捕获并响应潜在的数据库错误。
  5. Mongoose 版本兼容性: 随着 Mongoose 版本的迭代,一些连接选项可能会被弃用或成为默认行为。请查阅 Mongoose 官方文档以获取最新和推荐的配置。例如,useCreateIndex, useFindAndModify, useNewUrlParser, useUnifiedTopology 等选项在 Mongoose 6.0 及更高版本中已不再需要显式设置。
  6. 代码组织: 对于大型应用,可以考虑将每个数据库连接及其相关的模型定义组织到单独的文件或模块中,提高代码的可维护性。

通过遵循这些指南,你将能够有效地在 Mongoose 中管理和使用多个数据库连接及其对应的模型,从而构建健壮且可扩展的 Node.js 应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
scripterror怎么解决
scripterror怎么解决

scripterror的解决办法有检查语法、文件路径、检查网络连接、浏览器兼容性、使用try-catch语句、使用开发者工具进行调试、更新浏览器和JavaScript库或寻求专业帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

208

2023.10.18

500error怎么解决
500error怎么解决

500error的解决办法有检查服务器日志、检查代码、检查服务器配置、更新软件版本、重新启动服务、调试代码和寻求帮助等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

295

2023.10.25

c语言const用法
c语言const用法

const是关键字,可以用于声明常量、函数参数中的const修饰符、const修饰函数返回值、const修饰指针。详细介绍:1、声明常量,const关键字可用于声明常量,常量的值在程序运行期间不可修改,常量可以是基本数据类型,如整数、浮点数、字符等,也可是自定义的数据类型;2、函数参数中的const修饰符,const关键字可用于函数的参数中,表示该参数在函数内部不可修改等等。

530

2023.09.20

js正则表达式
js正则表达式

php中文网为大家提供各种js正则表达式语法大全以及各种js正则表达式使用的方法,还有更多js正则表达式的相关文章、相关下载、相关课程,供大家免费下载体验。

512

2023.06.20

js获取当前时间
js获取当前时间

JS全称JavaScript,是一种具有函数优先的轻量级,解释型或即时编译型的编程语言;它是一种属于网络的高级脚本语言,主要用于Web,常用来为网页添加各式各样的动态功能。js怎么获取当前时间呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

244

2023.07.28

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js是什么意思
js是什么意思

JS是JavaScript的缩写,它是一种广泛应用于网页开发的脚本语言。JavaScript是一种解释性的、基于对象和事件驱动的编程语言,通常用于为网页增加交互性和动态性。它可以在网页上实现复杂的功能和效果,如表单验证、页面元素操作、动画效果、数据交互等。

5306

2023.08.17

js删除节点的方法
js删除节点的方法

js删除节点的方法有:1、removeChild()方法,用于从父节点中移除指定的子节点,它需要两个参数,第一个参数是要删除的子节点,第二个参数是父节点;2、parentNode.removeChild()方法,可以直接通过父节点调用来删除子节点;3、remove()方法,可以直接删除节点,而无需指定父节点;4、innerHTML属性,用于删除节点的内容。

481

2023.09.01

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
MongoDB 教程
MongoDB 教程

共17课时 | 2.4万人学习

黑马云课堂mongodb实操视频教程
黑马云课堂mongodb实操视频教程

共11课时 | 3.1万人学习

MongoDB 教程
MongoDB 教程

共42课时 | 27.2万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号