
在复杂的应用场景中,我们可能需要连接到多个 mongodb 数据库,例如,一个用于主业务数据,另一个用于日志或用户配置。mongoose 提供了灵活的机制来管理这些独立的数据库连接。
Mongoose 主要有两种建立连接的方式:
理解这两种连接方式的区别至关重要。当您使用 createConnection 创建一个新连接时,您在此连接上定义的所有模型都将专门绑定到该连接实例,从而实现数据操作的隔离性。
使用 mongoose.createConnection() 是建立独立数据库连接的核心。它接受与 mongoose.connect() 相同的连接字符串和选项对象。
const mongoose = require('mongoose');
// 定义通用的连接选项,确保连接的稳定性和兼容性
const connectionOptions = {
useCreateIndex: true, // 确保创建索引时兼容性
useNewUrlParser: true, // 使用新的URL解析器
useUnifiedTopology: true, // 使用新的统一拓扑引擎
useFindAndModify: false // 禁用 findAndModify(),使用 findOneAndUpdate() 或 findOneAndRemove()
};
// 建立第一个数据库连接(例如,主数据库)
const dbMain = mongoose.createConnection("mongodb://localhost/db_main", connectionOptions);
// 监听连接事件,以便了解连接状态
dbMain.on('connected', () => {
console.log('成功连接到主数据库: db_main');
});
dbMain.on('error', (err) => {
console.error('主数据库连接错误:', err);
});
dbMain.on('disconnected', () => {
console.log('主数据库连接已断开: db_main');
});
// 建立第二个数据库连接(例如,db_en 数据库)
const dbEn = mongoose.createConnection("mongodb://localhost/db_en", connectionOptions);
dbEn.on('connected', () => {
console.log('成功连接到第二数据库: db_en');
});
dbEn.on('error', (err) => {
console.error('第二数据库连接错误:', err);
});
dbEn.on('disconnected', () => {
console.log('第二数据库连接已断开: db_en');
});
// 您可以根据需要建立更多连接一旦建立了独立的连接实例(例如 dbEn),您就可以在该连接上定义和使用模型。关键在于使用 connection.model() 方法。这个方法的作用是:
重要提示: connection.model(name, schema) 方法的返回值是该模型的一个构造函数。您必须将这个返回值赋给一个变量,然后通过这个变量来实例化模型。直接尝试 new connection.ModelName() 是错误的,因为 connection.ModelName 并不是一个可用的属性。
让我们通过一个示例来理解这一点,并解释常见的错误:
// 定义一个 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 }
}
});
// 在 dbEn 连接上定义 Price 模型
// 正确的做法:将 dbEn.model() 的返回值赋给一个变量
const PriceModelEn = dbEn.model('Price', priceSchema);
// 错误示范:直接尝试从连接对象访问模型,会导致 TypeError
// dbEn.model('Price', priceSchema); // 此时模型已在 dbEn 上定义,但没有赋值给变量
// const ggg = new dbEn.Price(); // 错误!dbEn.Price 不存在,dbEn 实例上没有名为 Price 的属性
// 错误信息:TypeError: conn.Price is not a constructor错误分析: 原始问题中出现的 TypeError: conn.Price is not a constructor 正是由于混淆了 connection.model() 的用法。conn.model('Price', priceSchema) 确实在 conn 连接上注册了 Price 模型,但它并不会在 conn 对象上创建一个名为 Price 的直接可访问属性。您需要捕获 conn.model() 的返回值,该返回值就是模型的构造函数。
一旦您正确地获取了特定连接上的模型构造函数(例如 PriceModelEn),就可以像使用普通 Mongoose 模型一样来实例化和操作它。
// ... (承接上面的连接建立和 Schema 定义) ...
// 异步函数用于演示数据操作
async function savePriceData() {
try {
// 使用 PriceModelEn 构造函数来创建文档实例
const newPrice = new PriceModelEn({
fixed: {
1: 299,
3: 599
}
});
// 保存文档到 db_en 数据库
const savedPrice = await newPrice.save();
console.log('价格数据成功保存到 db_en 数据库:', savedPrice);
// 您也可以通过模型直接查询数据
const foundPrice = await PriceModelEn.findOne({ 'fixed.1': 299 });
console.log('从 db_en 数据库查询到的价格数据:', foundPrice);
} catch (error) {
console.error('保存或查询价格数据时发生错误:', error);
} finally {
// 在操作完成后,可以选择关闭连接
// dbEn.close();
// dbMain.close(); // 如果不再需要,也关闭主连接
}
}
// 调用函数执行数据操作
savePriceData();
// 示例:如果在 dbMain 上也有一个 User 模型
// const userSchema = new mongoose.Schema({ name: String });
// const UserModelMain = dbMain.model('User', userSchema);
// async function saveUserData() {
// try {
// const newUser = new UserModelMain({ name: 'Alice' });
// await newUser.save();
// console.log('用户数据成功保存到 db_main 数据库:', newUser);
// } catch (error) {
// console.error('保存用户数据时发生错误:', error);
// }
// }
// saveUserData();通过 mongoose.createConnection(),Mongoose 提供了强大的多数据库连接管理能力,允许您为不同的业务或数据类型隔离数据存储。关键在于理解 connection.model() 方法的正确用法:它返回的是模型构造函数,必须将其赋值给一个变量才能用于实例化。遵循这些最佳实践,您可以有效地在 Mongoose 应用中管理和操作多个数据库。
以上就是Mongoose 多数据库连接与模型管理深度指南的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号