能否在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资源 数组中的populate 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()的工作方式以及包含模型中引用的定义非常重要.既然你真的只能指向"一个"模型作为参考,但是还有一些其他的魔法可以让一个模型看起来一样多.

示例 list :

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"
  }
]

您还可以取消对 list 中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值.

因此,虽然存储在一个集合中,但它可能看起来是多个集合,但也有将它们放在一起以进行其他有用操作的好处.这就是你如何应用你正在寻找的"多态性".

Mongodb相关问答推荐

MongoDB通过查找具有多个数组的对象进行聚合

除非满足某个条件,否则Mongo是否按日期排序?

聚合 $accumulator + $project

通过 docker 运行的 MongoDB 服务器无法互相看到(名称解析中的临时故障)

MongoDB 存储大量指标/分析数据的方法

Mongo 删除最后的文件

MongoDB C# 驱动程序 - 如何将 _id 存储为 ObjectId 但映射到字符串 Id 属性?

哪个库最适合用于带有 Scala 的 MongoDB?

Spring Boot MongoDB 连接问题

将 MongoDB 地理空间索引与 3d 数据结合使用

oplog 在独立 mongod 上启用,不适用于副本集

在 MongoDB 中创建简短、唯一的对象 ID

mongodb,pymongo,aggregate聚合给出奇怪的输出

MongoDB - 清除嵌套数组中的元素

MongoDb:如何将附加对象插入对象集合?

使用 Spring Security + Spring 数据 + MongoDB 进行身份验证

使用自定义 _id 值时 mongodb 中的 Upserts

MongoDB 的 MMAPV1、WiredTiger 或 In-Memory StorageEngine 如何 Select ?

spring 数据MongoDB.生成id的错误

MongoDB 引用的最佳实践