在MongoDB中,是否可以使用来自另一个字段的值更新一个字段的值?等价的SQL是这样的:
UPDATE Person SET Name = FirstName + ' ' + LastName
MongoDB的伪代码是:
db.person.update( {}, { $set : { name : firstName + ' ' + lastName } );
在MongoDB中,是否可以使用来自另一个字段的值更新一个字段的值?等价的SQL是这样的:
UPDATE Person SET Name = FirstName + ' ' + LastName
MongoDB的伪代码是:
db.person.update( {}, { $set : { name : firstName + ' ' + lastName } );
当前回答
MongoDB 4.2+ Golang
result, err := collection.UpdateMany(ctx, bson.M{},
mongo.Pipeline{
bson.D{{"$set",
bson.M{"name": bson.M{"$concat": []string{"$lastName", " ", "$firstName"}}}
}},
)
其他回答
MongoDB 4.2+ Golang
result, err := collection.UpdateMany(ctx, bson.M{},
mongo.Pipeline{
bson.D{{"$set",
bson.M{"name": bson.M{"$concat": []string{"$lastName", " ", "$firstName"}}}
}},
)
从Mongo 4.2开始,db.collection.update()可以接受一个聚合管道,最终允许基于另一个字段更新/创建一个字段:
// { firstName: "Hello", lastName: "World" }
db.collection.updateMany(
{},
[{ $set: { name: { $concat: [ "$firstName", " ", "$lastName" ] } } }]
)
// { "firstName" : "Hello", "lastName" : "World", "name" : "Hello World" }
第一部分{}是匹配查询,过滤要更新的文档(在本例中是所有文档)。 第二部分[{$set: {name:{…}}]是更新聚合管道(注意方括号表示使用聚合管道)。$set是一个新的聚合操作符,是$addFields的别名。
Update()方法将聚合管道作为参数,如
db.collection_name.update(
{
// Query
},
[
// Aggregation pipeline
{ "$set": { "id": "$_id" } }
],
{
// Options
"multi": true // false when a single doc has to be updated
}
)
可以使用聚合管道使用现有值设置或取消设置字段。
注意:使用带字段名的$来指定要读取的字段。
下面是我们针对~150_000条记录将一个字段复制到另一个字段的方法。它花了大约6分钟,但与实例化和遍历相同数量的ruby对象相比,仍然明显减少了资源消耗。
js_query = %({
$or : [
{
'settings.mobile_notifications' : { $exists : false },
'settings.mobile_admin_notifications' : { $exists : false }
}
]
})
js_for_each = %(function(user) {
if (!user.settings.hasOwnProperty('mobile_notifications')) {
user.settings.mobile_notifications = user.settings.email_notifications;
}
if (!user.settings.hasOwnProperty('mobile_admin_notifications')) {
user.settings.mobile_admin_notifications = user.settings.email_admin_notifications;
}
db.users.save(user);
})
js = "db.users.find(#{js_query}).forEach(#{js_for_each});"
Mongoid::Sessions.default.command('$eval' => js)
你应该迭代。针对您的具体情况:
db.person.find().snapshot().forEach(
function (elem) {
db.person.update(
{
_id: elem._id
},
{
$set: {
name: elem.firstname + ' ' + elem.lastname
}
}
);
}
);