【发布时间】:2016-07-13 00:22:49
【问题描述】:
我有一个来自 mongoose 的文档,我想在 JSON 编码之前扩展它并作为响应发送出去。如果我尝试向文档添加属性,它将被忽略。 Object.getOwnPropertyNames(doc) 中没有出现这些属性,因此无法进行正常扩展。奇怪的是JSON.parse(JSON.encode(doc)) 工作并返回具有所有正确属性的对象。有没有更好的方法来做到这一点?
【问题讨论】:
我有一个来自 mongoose 的文档,我想在 JSON 编码之前扩展它并作为响应发送出去。如果我尝试向文档添加属性,它将被忽略。 Object.getOwnPropertyNames(doc) 中没有出现这些属性,因此无法进行正常扩展。奇怪的是JSON.parse(JSON.encode(doc)) 工作并返回具有所有正确属性的对象。有没有更好的方法来做到这一点?
【问题讨论】:
Mongoose Models 继承自 Documents,它有一个 toObject() 方法。我相信你要找的应该是doc.toObject()的结果。
http://mongoosejs.com/docs/api.html#document_Document-toObject
【讨论】:
Model.find({}) 上返回数组,这是否有效,文档返回的是一个数组。你能 docs.toObject 吗?
var docArray = modelArray.map(function(model) { return model.toObject(); });
另一种方法是通过在查询链中使用lean() 告诉Mongoose,您需要的只是返回文档的纯JavaScript 版本。这样 Mongoose 就跳过了创建完整模型实例的步骤,您可以直接得到一个 doc ,您可以修改:
MyModel.findOne().lean().exec(function(err, doc) {
doc.addedProperty = 'foobar';
res.json(doc);
});
【讨论】:
lean 通常性能更高,因为您跳过了首先创建完整 Mongoose 文档的开销。
lean() 会在查询实际执行之前设置选项(通过exec),以便结果直接采用“精益”形式。请参阅this question,了解调用lean 对性能的积极影响。
如果属性不在模型中的快速方法:
document.set( key,value, { strict: false });
【讨论】:
在某些情况下,正如@JohnnyHK 所建议的那样,您可能希望将对象作为纯 Javascript 获取。 如Mongoose Documentation 中所述,还有另一种方法可以直接将数据作为对象进行查询:
const docs = await Model.find().lean();
另外如果有人想有条件地转向一个对象,也可以作为option参数,见第三个参数find() docs:
const toObject = true;
const docs = await Model.find({},null,{lean:toObject});
可用于以下功能:find()、findOne()、findById()、findOneAndUpdate() 和 findByIdAndUpdate()。
还值得一提的是,_id 属性不是一个字符串对象,就像你会做 JSON.parse(JSON.stringify(object)) 一样,而是一个来自猫鼬类型的 ObjectId,所以在将它与字符串进行比较时,将其转换为之前的字符串:String(object._id) === otherStringId
【讨论】:
解决此类问题的更好方法是像这样使用doc.toObject()
doc.toObject({ getters: true })
其他选项包括:
getters: 应用所有 getter(路径和虚拟 getter)virtuals: 应用虚拟吸气剂(可以覆盖吸气剂选项)minimize:删除空对象(默认为true)transform: 在返回之前应用于结果文档的转换函数depopulate: 删除所有填充的路径,用它们的原始引用替换它们(默认为 false)versionKey:是否包含版本密钥(默认为true)例如你可以说
Model.findOne().exec((err, doc) => {
if (!err) {
doc.toObject({ getters: true })
console.log('doc _id:', doc._id)
}
})
现在它可以工作了。
参考见:http://mongoosejs.com/docs/api.html#document_Document-toObject
【讨论】:
为了从 Mongoose 文档中获取普通对象,我使用了 _doc 属性,如下所示
mongooseDoc._doc //returns plain json object
我尝试使用toObject,但它给了我函数、参数和所有其他我不需要的东西。
【讨论】:
_ 访问属性或方法不是理想的解决方案。使用精益可能会解决您的问题。
_doc 属性而没有遇到任何问题。使用lean 方法可以为您提供无法执行猫鼬模型操作的普通对象。
lean 选项告诉 Mongoose 跳过水合结果文档。这使得查询更快,内存占用更少,但结果文档是普通的旧 JavaScript 对象 (POJO),而不是 Mongoose 文档。
const leanDoc = await MyModel.findOne().lean();
不必使用 JSON.parse() 方法
【讨论】:
您还可以将对象字符串化,然后再次解析以生成普通对象。 例如:-
const obj = JSON.parse(JSON.stringify(mongoObj))
【讨论】:
_id 作为字符串获取的优势。