MongoDB聚合运算符:$setField
文章目录
- MongoDB聚合运算符:$setField
- 语法
- 字段说明
- 使用
- 举例
- 添加包含点号 (.) 的字段
- 添加以($)开头的字段
- 更新包含(.)的字段
- 更新以($)开头的字段
- 删除包含(.)的字段
- 删除以($)开头的字段
$setField
聚合运算符用于添加、更新、删除文档中的指定字段,可以用于字段名称包含
.
或以
$
开头的字段。从5.0版本开始支持。
语法
{$setField: {field: <String>,input: <Object>,value: <Expression>}
}
字段说明
字段 | 类型 | 说明 |
---|---|---|
field | 字符串 | 要添加、更新或删除的输入对象中的字段,字段可以是字符串表达式 |
input | 对象 | 指定要更新的文档,input 必须是可以解析为文档的对象、缺失、空或未定义 |
value | 表达式 | 要分配给field 的值,value 可以是任何有效的表达式,如果设置为$$REMOVE ,则会从输入文档中删除字段 |
使用
-
如果
input
的计算结果为缺失、undefined
或null
,则$setField
返回null
并且不更新input
。 -
如果
input
的计算结果为除对象、缺失、undefined
或null
之外的任何内容,则$setField
将返回错误。 -
如果
field
解析为字符串常量以外的任何内容,则$setField
返回错误。 -
如果
input
中不存在field
,$setField
将添加一个field
。 -
$setField
不会隐式遍历对象或数组,例如$setField
将字段值"a.b.c"
计算为顶级字段"a.b.c"
,而不是嵌套字段{ "a": { "b": { "c": } } }
。 -
$unsetField
是输入值为$$REMOVE
的$setField
的别名。以下表达式是等效的:{$setField: {field: <field name>,input: “$$ROOT”,value: "$$REMOVE"}}{$unsetField: {field: <field name>,input: “$$ROOT”}}
举例
添加包含点号 (.) 的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany( [{ "_id" : 1, "item" : "sweatshirt", price: 45.99, qty: 300 }{ "_id" : 2, "item" : "winter coat", price: 499.99, qty: 200 }{ "_id" : 3, "item" : "sun dress", price: 199.99, qty: 250 }{ "_id" : 4, "item" : "leather boots", price: 249.99, qty: 300 }{ "_id" : 5, "item" : "bow tie", price: 9.99, qty: 180 }
] )
下面的聚合操作使用 $replaceWith
管道阶段和 $setField
操作符为每个文档添加一个新字段 "price.usd"
。"price.usd"
的值将等于每个文档 "price"
的值。最后,使用 $unset
管道阶段删除 "price"
字段:
db.inventory.aggregate( [{ $replaceWith: {$setField: {field: "price.usd",input: "$$ROOT",value: "$price"} } },{ $unset: "price" }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300, 'price.usd': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, 'price.usd': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, 'price.usd': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, 'price.usd': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, 'price.usd': 9.99 }
]
添加以($)开头的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany( [{ "_id" : 1, "item" : "sweatshirt", price: 45.99, qty: 300 }{ "_id" : 2, "item" : "winter coat", price: 499.99, qty: 200 }{ "_id" : 3, "item" : "sun dress", price: 199.99, qty: 250 }{ "_id" : 4, "item" : "leather boots", price: 249.99, qty: 300 }{ "_id" : 5, "item" : "bow tie", price: 9.99, qty: 180 }
] )
以下聚合操作使用 $replaceWith
管道阶段以及 $setField
和 $literal
运算符向每个文档添加新字段"$price"
,"$price"
的值等于每个文档中"price"
的值,最后使用 $unset
管道阶段删除"price"
字段:
db.inventory.aggregate( [{ $replaceWith: {$setField: {field: { $literal: "$price" },input: "$$ROOT",value: "$price"} } },{ $unset: "price" }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300, '$price': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, '$price': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, '$price': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, '$price': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, '$price': 9.99 }
]
更新包含(.)的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany( [{ _id: 1, item: 'sweatshirt', qty: 300, 'price.usd': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, 'price.usd': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, 'price.usd': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, 'price.usd': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, 'price.usd': 9.99 }
] )
下面的聚合操作使用 $match
管道阶段查找制定的文档,并使用 $replaceWith
管道阶段和 $setField
运算符更新匹配文档中的"price.usd"
字段:
db.inventory.aggregate( [{ $match: { _id: 1 } },{ $replaceWith: {$setField: {field: "price.usd",input: "$$ROOT",value: 49.99} } }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300, 'price.usd': 49.99 }
]
更新以($)开头的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany([{ _id: 1, item: 'sweatshirt', qty: 300, '$price': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, '$price': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, '$price': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, '$price': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, '$price': 9.99 }
] )
以下操作使用 $match
管道阶段来查找文档,并使用 $replaceWith
管道阶段以及 $setField
和 $literal
运算符来更新匹配文档中的"$price"
字段:
db.inventory.aggregate( [{ $match: { _id: 1 } },{ $replaceWith: {$setField: {field: { $literal: "$price" },input: "$$ROOT",value: 49.99} } }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300, '$price': 49.99 }
]
删除包含(.)的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany([{ _id: 1, item: 'sweatshirt', qty: 300, 'price.usd': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, 'price.usd': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, 'price.usd': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, 'price.usd': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, 'price.usd': 9.99 }
] )
以下操作使用 $replaceWith
管道阶段、$setField
操作符和 $$REMOVE
从文档中删除 "price.usd"
字段::
db.inventory.aggregate( [{ $replaceWith: {$setField: {field: "price.usd",input: "$$ROOT",value: "$$REMOVE"} } }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300 },{ _id: 2, item: 'winter coat', qty: 200 },{ _id: 3, item: 'sun dress', qty: 250 },{ _id: 4, item: 'leather boots', qty: 300 },{ _id: 5, item: 'bow tie', qty: 180 }
]
使用 $unsetField
也可以完成查询返回相同的结果:
db.inventory.aggregate( [{ $replaceWith: {$unsetField: {field: "price.usd",input: "$$ROOT"} } }
] )
删除以($)开头的字段
使用下面的脚本创建inventory
集合:
db.inventory.insertMany( [{ _id: 1, item: 'sweatshirt', qty: 300, '$price': 45.99 },{ _id: 2, item: 'winter coat', qty: 200, '$price': 499.99 },{ _id: 3, item: 'sun dress', qty: 250, '$price': 199.99 },{ _id: 4, item: 'leather boots', qty: 300, '$price': 249.99 },{ _id: 5, item: 'bow tie', qty: 180, '$price': 9.99 }
} )
以下操作使用 $replaceWith
管道阶段、$setField
和 $literal
运算符以及 $$REMOVE
删除文档中的"$price"
字段:
db.inventory.aggregate( [{ $replaceWith: {$setField: {field: { $literal: "$price" },input: "$$ROOT",value: "$$REMOVE"} } }
] )
操作返回下面的结果:
[{ _id: 1, item: 'sweatshirt', qty: 300 },{ _id: 2, item: 'winter coat', qty: 200 },{ _id: 3, item: 'sun dress', qty: 250 },{ _id: 4, item: 'leather boots', qty: 300 },{ _id: 5, item: 'bow tie', qty: 180 }
]
使用 $unsetField
可以实现相同的效果:
db.inventory.aggregate( [{ $replaceWith: {$unsetField: {field: { $literal: "$price" },input: "$$ROOT"} } }
] )