$replaceRoot (aggregation)

On this page本页内容

Definition定义

$replaceRoot

New in version 3.4.版本3.4中的新功能。

Replaces the input document with the specified document. 用指定的文档替换输入文档。The operation replaces all existing fields in the input document, including the _id field. 该操作将替换输入文档中的所有现有字段,包括_id字段。You can promote an existing embedded document to the top level, or create a new document for promotion (see example).您可以将现有的嵌入式文档升级到顶层,或者创建一个新文档进行升级(参见示例)。

Note

Starting in version 4.2, MongoDB adds a new aggregation pipeline stage $replaceWith that is an alias for $replaceRoot.从4.2版开始,MongoDB添加了一个新的聚合管道阶段$replaceWith,它是$replaceRoot的别名。

The $replaceRoot stage has the following form:$replaceRoot阶段具有以下形式:

{ $replaceRoot: { newRoot: <replacementDocument> } }

The replacement document can be any valid expression that resolves to a document. 替换文档可以是解析为文档的任何有效表达式The stage errors and fails if <replacementDocument> is not a document. 如果<replacementDocument>不是文档,则阶段会出错并失败。For more information on expressions, see Expressions.有关表达式的详细信息,请参阅表达式

Behavior行为

If the <replacementDocument> is not a document, $replaceRoot errors and fails.如果<replacementDocument>不是文档,$replaceRoot会出错并失败。

If the <replacementDocument> resolves to a missing document (i.e. the document does not exist), $replaceRoot errors and fails. 如果<replacementDocument>解析为缺少的文档(即该文档不存在),$replaceRoot将出错并失败。For example, create a collection with the following documents:例如,创建包含以下文档的集合:

db.collection.insertMany([
   { "_id": 1, "name" : { "first" : "John", "last" : "Backus" } },
   { "_id": 2, "name" : { "first" : "John", "last" : "McCarthy" } },
   { "_id": 3, "name": { "first" : "Grace", "last" : "Hopper" } },
   { "_id": 4, "firstname": "Ole-Johan", "lastname" : "Dahl" },
])

Then the following $replaceRoot operation fails because one of the documents does not have the name field:然后,以下$replaceRoot操作失败,因为其中一个文档没有name字段:

db.collection.aggregate([
   { $replaceRoot: { newRoot: "$name" } }
])

To avoid the error, you can use $mergeObjects to merge the name document into some default document; for example:为了避免错误,可以使用$mergeObjects将名称文档合并到一些默认文档中;例如:

db.collection.aggregate([
   { $replaceRoot: { newRoot: { $mergeObjects: [ { _id: "$_id", first: "", last: "" }, "$name" ] } } }
])

Alternatively, you can skip the documents that are missing the name field by including a $match stage to check for existence of the document field before passing documents to the $replaceRoot stage:或者,在将文档传递到$replaceRoot阶段之前,可以通过包含$match阶段来检查文档字段是否存在,从而跳过缺少name字段的文档:

db.collection.aggregate([
   { $match: { name : { $exists: true, $not: { $type: "array" }, $type: "object" } } },
   { $replaceRoot: { newRoot: "$name" } }
])

Or, you can use $ifNull expression to specify some other document to be root; for example:或者,您可以使用$ifNull表达式指定其他文档作为根;例如:

db.collection.aggregate([
   { $replaceRoot: { newRoot: { $ifNull: [ "$name", { _id: "$_id", missingName: true} ] } } }
])

Examples示例

$replaceRoot with an Embedded Document Field带有嵌入式文档字段

A collection named people contains the following documents:名为people的集合包含以下文档:

{ "_id" : 1, "name" : "Arlene", "age" : 34, "pets" : { "dogs" : 2, "cats" : 1 } }
{ "_id" : 2, "name" : "Sam", "age" : 41, "pets" : { "cats" : 1, "fish" : 3 } }
{ "_id" : 3, "name" : "Maria", "age" : 25 }

The following operation uses the $replaceRoot stage to replace each input document with the result of a $mergeObjects operation. 以下操作使用$replaceRoot阶段将每个输入文档替换为$mergeObjects操作的结果。The $mergeObjects expression merges the specified default document with the pets document.$mergeObjects表达式将指定的默认文档与pets文档合并。

db.people.aggregate( [
   { $replaceRoot: { newRoot: { $mergeObjects:  [ { dogs: 0, cats: 0, birds: 0, fish: 0 }, "$pets" ] }} }
] )

The operation returns the following results:操作返回以下结果:

{ "dogs" : 2, "cats" : 1, "birds" : 0, "fish" : 0 }
{ "dogs" : 0, "cats" : 1, "birds" : 0, "fish" : 3 }
{ "dogs" : 0, "cats" : 0, "birds" : 0, "fish" : 0 }

$replaceRoot with a Document Nested in an Array带有嵌套在数组中的文档

A collection named students contains the following documents:名为students的集合包含以下文档:

db.students.insertMany([
   {
      "_id" : 1,
      "grades" : [
         { "test": 1, "grade" : 80, "mean" : 75, "std" : 6 },
         { "test": 2, "grade" : 85, "mean" : 90, "std" : 4 },
         { "test": 3, "grade" : 95, "mean" : 85, "std" : 6 }
      ]
   },
   {
      "_id" : 2,
      "grades" : [
         { "test": 1, "grade" : 90, "mean" : 75, "std" : 6 },
         { "test": 2, "grade" : 87, "mean" : 90, "std" : 3 },
         { "test": 3, "grade" : 91, "mean" : 85, "std" : 4 }
      ]
   }
])

The following operation promotes the embedded document(s) with the grade field greater than or equal to 90 to the top level:以下操作将grade字段大于或等于90的嵌入文档提升到顶层:

db.students.aggregate( [
   { $unwind: "$grades" },
   { $match: { "grades.grade" : { $gte: 90 } } },
   { $replaceRoot: { newRoot: "$grades" } }
] )

The operation returns the following results:操作返回以下结果:

{ "test" : 3, "grade" : 95, "mean" : 85, "std" : 6 }
{ "test" : 1, "grade" : 90, "mean" : 75, "std" : 6 }
{ "test" : 3, "grade" : 91, "mean" : 85, "std" : 4 }

$replaceRoot with a newly created document使用新创建的文档

You can also create new documents as part of the $replaceRoot stage and use them to replace all the other fields.您还可以创建新文档作为$replaceRoot阶段的一部分,并使用它们替换所有其他字段。

A collection named contacts contains the following documents:名为contacts的集合包含以下文档:

{ "_id" : 1, "first_name" : "Gary", "last_name" : "Sheffield", "city" : "New York" }
{ "_id" : 2, "first_name" : "Nancy", "last_name" : "Walker", "city" : "Anaheim" }
{ "_id" : 3, "first_name" : "Peter", "last_name" : "Sumner", "city" : "Toledo" }

The following operation creates a new document out of the first_name and last_name fields.下面的操作使用first_namelast_name字段创建一个新文档。

db.contacts.aggregate( [
   {
      $replaceRoot: {
         newRoot: {
            full_name: {
               $concat : [ "$first_name", " ", "$last_name" ]
            }
         }
      }
   }
] )

The operation returns the following results:操作返回以下结果:

{ "full_name" : "Gary Sheffield" }
{ "full_name" : "Nancy Walker" }
{ "full_name" : "Peter Sumner" }

$replaceRoot with a New Document Created from $$ROOT and a Default Document$replaceRoot,其中包含从$$ROOT创建的新文档和默认文档

Create a collection named contacts with the following documents:使用以下文档创建名为contacts的集合:

db.contacts.insert([
   { "_id" : 1, name: "Fred", email: "fred@example.net" },
   { "_id" : 2, name: "Frank N. Stine", cell: "012-345-9999" },
   { "_id" : 3, name: "Gren Dell", home: "987-654-3210", email: "beo@example.net" }
]);

The following operation uses $replaceRoot with $mergeObjects to output current documents with default values for missing fields:以下操作使用$replaceRoot$mergeObjects来输出当前文档,其中包含缺失字段的默认值:

db.contacts.aggregate([
   { $replaceRoot: { newRoot: { $mergeObjects: [ { _id: "", name: "", email: "", cell: "", home: "" }, "$$ROOT" ] } } }
])

The aggregation returns the following documents:聚合将返回以下文档:

{ "_id" : 1, "name" : "Fred", "email" : "fred@example.net", "cell" : "", "home" : "" }
{ "_id" : 2, "name" : "Frank N. Stine", "email" : "", "cell" : "012-345-9999", "home" : "" }
{ "_id" : 3, "name" : "Gren Dell", "email" : "beo@example.net", "cell" : "", "home" : "987-654-3210" }