
在mongoose中,与mongodb数据库的交互围绕着“连接”(connection)和“模型”(model)两个核心概念展开。
连接 (Connection):代表了应用程序与MongoDB数据库之间的一个会话。
模型 (Model):是基于Schema定义的一个构造函数,用于创建和读取数据库中的文档。模型是与特定连接关联的。
当使用mongoose.createConnection()创建一个新的连接实例(例如conn)后,你需要通过该连接实例来定义和访问模型,而不是通过全局的mongoose对象或直接将模型视为连接对象的属性。
这个错误信息TypeError: conn.Price is not a constructor表明你尝试将conn.Price作为一个构造函数来使用,但它实际上并不是。
其根本原因在于:当你通过conn.model('Price', priceSchema)在特定连接conn上注册一个模型时,Mongoose并不会自动将这个模型构造函数挂载到conn对象的属性上,例如conn.Price。相反,conn.model()方法本身会返回这个已注册的模型构造函数。你需要将这个返回的构造函数赋值给一个变量,然后通过这个变量来实例化文档。
错误的示例代码:
const mongoose = require('mongoose');
const connectionOptions = { useNewUrlParser: true, useUnifiedTopology: true };
const conn = mongoose.createConnection("mongodb://localhost/db_en", connectionOptions);
const Schema = mongoose.Schema;
const priceSchema = new Schema({
fixed: {
1: { type: Number, default: 199 },
3: { type: Number, default: 499 },
6: { type: Number, default: 729 },
12: { type: Number, default: 999 }
}
});
// 在 conn 连接上注册模型,但没有捕获返回的模型构造函数
conn.model('Price', priceSchema);
// 错误:尝试将 conn.Price 作为构造函数使用
// const ggg = new conn.Price(); // 会抛出 TypeError为了解决上述错误并正确地在特定Mongoose连接上操作模型,你需要遵循以下步骤:
以下是完整的正确示例代码:
const mongoose = require('mongoose');
// 1. 建立独立连接
const connectionOptions = {
useCreateIndex: true, // 现代Mongoose版本可能不再需要此选项,或已弃用
useNewUrlParser: true,
useUnifiedTopology: true,
useFindAndModify: false // 现代Mongoose版本可能不再需要此选项,或已弃用
};
const conn = mongoose.createConnection("mongodb://localhost/db_en", connectionOptions);
// 监听连接事件,确保连接成功
conn.on('connected', () => {
console.log('MongoDB db_en connected successfully!');
});
conn.on('error', (err) => {
console.error('MongoDB db_en connection error:', err);
});
conn.on('disconnected', () => {
console.log('MongoDB db_en disconnected.');
});
// 2. 定义Schema
const Schema = mongoose.Schema;
const priceSchema = new Schema({
fixed: {
1: { type: Number, default: 199 },
3: { type: Number, default: 499 },
6: { type: Number, default: 729 },
12: { type: Number, default: 999 }
}
});
// 3. 在 conn 连接上注册模型,并捕获返回的模型构造函数
// PriceModel 现在是 Price 模型的构造函数,它与 conn 连接关联
const PriceModel = conn.model('Price', priceSchema);
// 4. 实例化和保存文档
async function createAndSavePrice() {
try {
// 使用 PriceModel 构造函数创建新文档实例
const newPrice = new PriceModel();
await newPrice.save();
console.log('Price document saved successfully:', newPrice);
} catch (error) {
console.error('Error saving price document:', error);
} finally {
// 在实际应用中,你可能不会立即关闭连接
// conn.close();
}
}
// 确保连接建立后再执行操作
conn.once('open', () => {
createAndSavePrice();
});
// 示例:从同一个连接中获取已定义的模型(如果已定义过)
async function findPrice() {
try {
// 如果 PriceModel 已经定义并赋值,可以直接使用
// 否则,可以通过 conn.model('Price') 获取已定义的模型
const retrievedPriceModel = conn.model('Price');
const prices = await retrievedPriceModel.find({});
console.log('Retrieved prices:', prices);
} catch (error) {
console.error('Error retrieving prices:', error);
}
}
conn.once('open', () => {
findPrice();
});理解这两者之间的区别对于正确管理Mongoose模型至关重要:
| 特性 | mongoose.model('Name', Schema) | connection.model('Name', Schema) |
|---|---|---|
| 关联连接 | 关联到 Mongoose 的默认连接 (通过 mongoose.connect() 建立)。 | 关联到特定的 connection 实例 (通过 mongoose.createConnection() 建立)。 |
| 何时使用 | 当应用程序只连接一个数据库时,或者你希望模型在所有默认连接操作中都可用时。 | 当应用程序需要连接多个数据库,并且希望每个模型只与特定的数据库连接关联时。 |
| 返回值 | 返回模型构造函数。 | 返回模型构造函数。 |
| 全局性 | 定义的模型是全局可访问的,可以通过 mongoose.model('Name') 在任何地方获取。 | 定义的模型只在该 connection 实例的上下文中注册和访问。 |
注意事项:
TypeError: conn.Price is not a constructor 错误是由于错误地尝试将Mongoose连接对象的一个非模型属性作为模型构造函数来使用。正确的做法是,当你在一个特定的连接实例上使用conn.model('ModelName', schema)定义模型时,务必将该方法返回的模型构造函数赋值给一个变量,然后通过这个变量来实例化你的文档。理解mongoose.model()和connection.model()之间的区别,并根据你的多数据库需求选择合适的方法,是高效利用Mongoose的关键。
以上就是Mongoose多数据库连接下的模型管理与实例化指南的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号