gpt4 book ai didi

javascript - 单个模式数组中的多个模式引用 - Mongoose

转载 作者:IT老高 更新时间:2023-10-28 13:11:37 25 4
gpt4 key购买 nike

您能否在 mongoose 模式中填充一个数组,并引用几个不同的模式选项?

为了澄清这个问题,假设我有以下架构:

var scenarioSchema = Schema({
_id : Number,
name : String,
guns : []
});

var ak47 = Schema({
_id : Number
//Bunch of AK specific parameters
});

var m16 = Schema({
_id : Number
//Bunch of M16 specific parameters
});

我可以用一堆 ak47 OR m16 填充枪阵列吗?我可以将 BOTH 放在同一个枪支阵列中吗?还是需要像这样在 assets 数组中填充 ref,从而将其限制为单个特定类型?

guns: [{ type: Schema.Types.ObjectId, ref: 'm16' }]

我知道我可以为不同的枪支类型设置单独的数组,但是随着项目的扩展,这会在架构中创建大量额外的字段,其中大部分会根据加载的场景留空。

var scenarioSchema = Schema({
_id : Number,
name : String,
ak47s : [{ type: Schema.Types.ObjectId, ref: 'ak47' }],
m16s: [{ type: Schema.Types.ObjectId, ref: 'm16' }]
});

回到这个问题,我可以在一个数组中粘贴多个架构引用吗?

最佳答案

你在这里寻找的是 Mongoose .discriminator()方法。这基本上允许您将不同类型的对象存储在同一个集合中,但将它们作为可区分的第一类对象。

请注意,这里的“相同集合”原则对于 .populate() 的工作方式以及包含模型中引用的定义很重要。因为无论如何你真的只能指向“一个”模型作为引用,但是还有一些其他的魔法可以让一个模型出现很多。

示例列表:

var util = require('util'),
async = require('async'),
mongoose = require('mongoose'),
Schema = mongoose.Schema;

mongoose.connect('mongodb://localhost/gunshow');

//mongoose.set("debug",true);

var scenarioSchema = new Schema({
"name": String,
"guns": [{ "type": Schema.Types.ObjectId, "ref": "Gun" }]
});

function BaseSchema() {
Schema.apply(this, arguments);

// Common Gun stuff
this.add({
"createdAt": { "type": Date, "default": Date.now }
});
}

util.inherits(BaseSchema, Schema);

var gunSchema = new BaseSchema();

var ak47Schema = new BaseSchema({
// Ak74 stuff
});

ak47Schema.methods.shoot = function() {
return "Crack!Crack";
};

var m16Schema = new BaseSchema({
// M16 Stuff
});

m16Schema.methods.shoot = function() {
return "Blam!!"
};


var Scenario = mongoose.model("Scenario", scenarioSchema);

var Gun = mongoose.model("Gun", gunSchema );
var Ak47 = Gun.discriminator("Ak47", ak47Schema );
var M16 = Gun.discriminator("M16", m16Schema );


async.series(
[
// Cleanup
function(callback) {
async.each([Scenario,Gun],function(model,callback) {
model.remove({},callback);
},callback);
},

// Add some guns and add to scenario
function(callback) {
async.waterfall(
[
function(callback) {
async.map([Ak47,M16],function(gun,callback) {
gun.create({},callback);
},callback);
},
function(guns,callback) {
Scenario.create({
"name": "Test",
"guns": guns
},callback);
}
],
callback
);
},

// Get populated scenario
function(callback) {
Scenario.findOne().populate("guns").exec(function(err,data) {

console.log("Populated:\n%s",JSON.stringify(data,undefined,2));

// Shoot each gun for fun!
data.guns.forEach(function(gun) {
console.log("%s says %s",gun.__t,gun.shoot());
});

callback(err);
});
},

// Show the Guns collection
function(callback) {
Gun.find().exec(function(err,guns) {
console.log("Guns:\n%s", JSON.stringify(guns,undefined,2));
callback(err);
});
},

// Show magic filtering
function(callback) {
Ak47.find().exec(function(err,ak47) {
console.log("Magic!:\n%s", JSON.stringify(ak47,undefined,2));
callback(err);
});
}
],
function(err) {
if (err) throw err;
mongoose.disconnect();
}
);

然后输出

Populated:
{
"_id": "56c508069d16fab84ead921d",
"name": "Test",
"__v": 0,
"guns": [
{
"_id": "56c508069d16fab84ead921b",
"__v": 0,
"__t": "Ak47",
"createdAt": "2016-02-17T23:53:42.853Z"
},
{
"_id": "56c508069d16fab84ead921c",
"__v": 0,
"__t": "M16",
"createdAt": "2016-02-17T23:53:42.862Z"
}
]
}
Ak47 says Crack!Crack
M16 says Blam!!
Guns:
[
{
"_id": "56c508069d16fab84ead921b",
"__v": 0,
"__t": "Ak47",
"createdAt": "2016-02-17T23:53:42.853Z"
},
{
"_id": "56c508069d16fab84ead921c",
"__v": 0,
"__t": "M16",
"createdAt": "2016-02-17T23:53:42.862Z"
}
]
Magic!:
[
{
"_id": "56c508069d16fab84ead921b",
"__v": 0,
"__t": "Ak47",
"createdAt": "2016-02-17T23:53:42.853Z"
}
]

您还可以取消注释列表中的 mongoose.set("debug",true) 行,以查看 mongoose 是如何实际构建调用的。

因此,这表明您可以将不同的模式应用于不同的第一类对象,甚至可以像真实对象一样附加不同的方法。 Mongoose 将这些都存储在附加模型的“枪”集合中,它将包含鉴别器引用的所有“类型”:

var Gun = mongoose.model("Gun", gunSchema );
var Ak47 = Gun.discriminator("Ak47", ak47Schema );
var M16 = Gun.discriminator("M16", m16Schema );

但每个不同的“类型”也以一种特殊的方式被它自己的模型引用。所以你会看到,当 mongoose 存储和读取对象时,有一个特殊的 __t 字段告诉它要应用哪个“模型”,从而附加模式。

作为一个示例,我们调用 .shoot() 方法,该方法对每个模型/模式都有不同的定义。而且您仍然可以将每个模型本身用作查询或其他操作的模型,因为 Ak47 将自动在所有查询/更新中应用 __t 值。

因此,尽管存储在一个集合中,但它看起来可能是多个集合,但也有利于将它们保持在一起以进行其他有用的操作。这就是您可以应用您正在寻找的“多态性”的方式。

关于javascript - 单个模式数组中的多个模式引用 - Mongoose ,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/35468855/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com