MongoDB:如何使用 Mongoose 添加或更新子文档?
Posted
技术标签:
【中文标题】MongoDB:如何使用 Mongoose 添加或更新子文档?【英文标题】:MongoDB: How to add or update a SubDocument with Mongoose? 【发布时间】:2016-01-22 19:47:21 【问题描述】:我正在尝试添加新的或根据现有子文档的存在来更新它。
给出以下架构:
var providerSchema = new Schema(
provider: String,
id: String,
accessToken: Object
)
var userSchema = new Schema(
email: type: String ,
provider: [providerSchema],
created_at: Date,
updated_at: Date
);
存储以下文档:
"_id":
"$oid": "5629f57b13bc066c17b88a0d"
,
"created_at":
"$date": "2015-10-23T08:53:15.684Z"
,
"updated_at":
"$date": "2015-10-23T08:53:15.684Z"
,
"email": "123123123@gmail.com",
"provider": [
"provider": "facebook",
"id": "123123",
"accessToken":
"expires_in": 123123,
"token_type": "bearer",
"access_token": "123123123"
,
"_id":
"$oid": "5629f57b13bc066c17b88a0e"
,
"accessToken":
"access_token": "123123",
"token_type": "bearer",
"uid": "123123"
,
"id": "123123",
"provider": "dropbox",
"_id":
"$oid": "5629f58813bc066c17b88a12"
],
"__v": 0
我正在尝试使用以下代码更新或添加新的子文档:
User.findOneAndUpdate(email: email,
"$push" :
"provider":
"provider": "dropbox",
"id": "123441212",
"accessToken": "sdasddad"
, function (error, doc)
....
);
很遗憾,现有的子文档没有更新,而是创建了新的子文档。
你能帮帮我吗?
提前致谢!
附:是否可以命名子文档?那么例如有一个名为“facebook”的子文档,可以通过名称直接访问?
--- 解决方案---
User.findOne(
"email": email)
.populate("provider")
.exec(function (err, data)
if(data === null && typeof data === "object")
// Create User
user.save(function (error)
if(error) return callback(new Error(error), null);
return callback(null, user);
)
else
//
if(err) return callback(new Error(err), null);
var prov = _.findWhere(data.provider, "provider": provider );
if(typeof variable_here === 'undefined')
data.provider.push(update);
else
_.extend(prov, update);
data.save(function (error)
if(error)
return callback(error, null);
else
return callback(null, data);
);
);
谢谢,侯赛因杰拉米!仍然有很多后备...
(一开始我在“_.findWhere”中挣扎;它是下划线;只需通过“npm install underscore --save”安装,然后将以下行放入文件“var _ = require('underscore');”)
【问题讨论】:
试试这个***.com/questions/26156687/… 【参考方案1】:考虑学生集合中的以下文档,其 Grades 元素值是嵌入文档的数组:
_id: 4,
grades: [
grade: 80, mean: 75, std: 8 ,
grade: 85, mean: 90, std: 5 ,
grade: 90, mean: 85, std: 3
]
使用位置$操作符更新嵌入文档中std字段的值,等级为85:
db.students.update(
_id: 4, "grades.grade": 85 ,
$set: "grades.$.std" : 6
)
下面的代码是完整的更新语法,如果您将 upsert 定义为 true,它将插入新记录,否则它将被更新。 默认值:假
db.collection.update(
<query>,
<update>,
upsert: <boolean>,
multi: <boolean>,
writeConcern: <document>
)
【讨论】:
谢谢阿洛克!如果 te SubDocument "grades.grade: 85" 不存在怎么办?它没有被创建。 代码已修改,请立即查看【参考方案2】:你可以试试这个。我认为这可能有效。
User.update(email: email,provider.provider:'dropbox',$set:"provider.$": "provider":
"provider": "dropbox",
"id": "123441212",
"accessToken": "sdasddad"
,function(error, doc)
...
);
【讨论】:
嗨 Ketha,非常感谢您的回答!仅当 provider.provider = dropbox 且 SubDocument 存在时,您的解决方案才有效。如果 SubDocument provider.provider = "facebook" 不存在但应该创建,则它不起作用。【参考方案3】:--- 解决方案---
User.findOne(
"email": email)
.populate("provider")
.exec(function (err, data)
if(data === null && typeof data === "object")
// Create User
user.save(function (error)
if(error) return callback(new Error(error), null);
return callback(null, user);
)
else
//
if(err) return callback(new Error(err), null);
var prov = _.findWhere(data.provider, "provider": provider );
if(typeof variable_here === 'undefined')
data.provider.push(update);
else
_.extend(prov, update);
data.save(function (error)
if(error)
return callback(error, null);
else
return callback(null, data);
);
);
谢谢,侯赛因杰拉米!仍然有很多后备...
(一开始我在“_.findWhere”中挣扎;它是下划线;只需通过“npm install underscore --save”安装,然后将以下行放入文件“var _ = require('underscore');”)
【讨论】:
以上是关于MongoDB:如何使用 Mongoose 添加或更新子文档?的主要内容,如果未能解决你的问题,请参考以下文章
哪个包更适合在真实项目 mongodb 或 mongoose 中使用?
如何使用 mongoose 或 mongodb 查询更改架构内的方案数组内的值。?
MongoDB:异步函数未完成或如何正确关闭与 mongoose 的连接
如何在 mongoose 或 mongodb 中制作数据模型