32

似乎 Mongoose 在内部做了一些非常时髦的事情。

var Foo = new mongoose.model('Foo', new mongoose.Schema({a: String, b: Number}));
var foo = new Foo({a: 'test'; b: 42}); 
var obj = {c: 1};
foo.goo = obj;                  // simple object assignment. obj should be 
                                  //   passed by reference to foo.goo. recall goo
                                  //   is not defined in the Foo model schema

console.log(foo.goo === obj);   // comparison directly after the assignment
    // => false, doesn't behave like normal JS object

本质上,每当您尝试处理 Mongoose 模型的属性时,

a) 在模型的模式中定义或

b) 定义为相同类型 (array, obj, ..) ...模型的行为甚至不像普通的 Javascript 对象。

将第4行切换为foo._doc.goo = obj控制台输出true

编辑:试图重现怪异

示例 1

 // Customer has a property 'name', but no property 'text'
 // I do this because I need to transform my data slightly before sending it
 // to client.
 models.Customer.find({}, function(err, data) {
     for (var i=0, len=data.length; i<len; ++i) {
        data[i] = data[i]._doc;            // if I don't do this, returned data
                                           // has no 'text' property
        data[i].text = data[i].name;       
    }
    res.json({success: err, response:data});
});
4

7 回答 7

9

更新

也许我误解了你原来的问题,但现在看起来你的问题的性质发生了变化,所以下面的信息不相关,但我要离开了。:)

我测试了你的代码,它对我来说很好。当您设置不属于架构的属性(或一些其他特殊属性)时,Mongoose 不会执行任何特殊代码。JavaScript 目前不支持调用尚不存在的属性的代码(goo例如,Mongoose 不能妨碍属性集)。

因此,当您设置属性时:

foo.goo = { c: 1 };

猫鼬没有参与。如果您console.log的代码不是您显示的代码,我可以看到它可能会错误地报告。

此外,当您send将结果返回为 JSON 时,JSON.stringify将调用toString您的 Mongoose 模型。发生这种情况时,Mongoose 仅​​使用架构上定义的属性。因此,默认情况下不会发回任何其他属性。您已经更改了data数组的性质,直接指向 Mongoose 数据,因此它避免了这个问题。

有关正常行为的详细信息

当您goo使用 Mongoose 设置属性时,会发生很多事情。Mongoose 通过Object.defineProperty(一些文档)创建属性 getter/setter。因此,当您设置goo定义为 a 的属性时,[String]会发生一些事情:

  1. 在将值设置到对象实例上之前调用 Mongoose 代码(与简单的 JavaScript 对象不同)
  2. Mongoose 创建一个数组(可选)来存储MongooseArray将包含数组数据的数据 (a)。在您提供的示例中,由于您没有传递数组,因此将创建它。
  3. Mongoose 将尝试将您的数据转换为正确的类型
  4. 它将调用toString作为转换的一部分传递的数据。

因此,结果是文档现在包含一个数组,其中包含toString您传递的对象的一个​​版本。

如果您检查了该goo属性的内容,您会看到它现在是一个包含单个元素的数组,该元素是一个包含[object Object]. 如果您选择了更基本的类型或匹配目标属性存储类型,您会看到基本的相等性检查会起作用。

于 2013-09-16T16:19:21.267 回答
8

_doc存在于 mongoose 对象上。

因为mongooseModel.findOne返回模型本身,所以模型具有结构(受保护的字段)。当您尝试使用 console.log 打印对象时,它只会为您提供数据库中的数据,因为 console.log 将打印对象的公共字段。

如果你尝试 JSON.stringify 之类的东西,那么你会看到猫鼬模型对象的内部。(_doc,状态...)

如果您想在对象中添加更多字段并且它不起作用

const car = model.findOne({_id:'1'})
car.someNewProp = true // this will not work

如果稍后您将属性设置为对象汽车,并且您之前没有在模型模式中指定,那么 Mongoose 模型将验证该字段是否存在以及它是否是有效类型。如果验证失败,则不会设置该属性。

于 2020-04-01T18:47:31.143 回答
2

有同样的问题。而不是更新我的模型。

const car = model.findOne({_id:'1'})
let temp = JSON.stringify(car);
let objCar = JSON.parse(temp);
objCar.color = 'Red'; //now add any property you want

这解决了我的问题

于 2021-04-13T10:14:44.873 回答
2

尝试使用精益

默认情况下,Mongoose 查询返回 Mongoose Document 类的一个实例。文档比普通的 JavaScript 对象重得多,因为它们有很多用于更改跟踪的内部状态。启用精益选项会告诉 Mongoose 跳过实例化完整的 Mongoose 文档,只给你 POJO。

https://mongoosejs.com/docs/tutorials/lean.html

于 2021-07-28T02:42:12.487 回答
1

您可以使用 toJSON() 而不是 _doc

于 2021-08-29T12:55:59.427 回答
1

我今天被困在这个问题上……把我逼疯了。不确定以下是否是一个好的解决方案(OP也提到了它),但这就是我克服这个问题的方法。

我的汽车对象:

 cars = [{"make" : "Toyota"}, {"make" : "Kia"}];

行动:

console.log("1. Cars before the color: " + car);

cars.forEach(function(car){
     car.colour = "Black";   //color is NOT defined in the model. 
});

console.log("2. Cars after the color: " + car);

有问题的控制台输出:

   1. Cars before the color: [{"make" : "Toyota"}, {"make" : "Kia"}];
   2. Cars after the color: [{"make" : "Toyota"}, {"make" : "Kia"}];   //No change! No new colour properties :(

如果您尝试通过 doc(例如car._doc .color = "black")传递模型中未定义的此属性,它将起作用(此颜色属性将分配给每辆汽车),但您不能似乎出于某种原因通过 EJS(前端)访问它。

解决方案:(同样,不确定这是否是最好的方法......但它对我有用):在汽车模型中添加这个新属性(颜色)。

var carSchema = mongoose.Schema({
   make: String,
   color: String   //New property.
})

重新定义模型后,一切正常/预期(不需要 _doc 'hacks' 等),我又活了一天;希望它可以帮助别人。

于 2019-01-27T09:20:08.357 回答
0

Mongoose 模型有些奇怪,您必须检查 Mongoose 是否已经在其模型数组中创建了模型。

这是我的解决方案:

import mongoose from 'mongoose';
createModel = (modelName="foo", schemaDef, schemaOptions = {})=> {
  const { Schema } = mongoose;
  const schema = Schema(schemaDef, schemaOptions);
  const Model = mongoose.models[modelName] || mongoose.model(modelName, schema);
  return Model;
}

我为我的模型使用我自己的猫鼬模型类和基类。我做了这个,它应该适合你。

于 2019-03-29T16:51:01.693 回答