$locf (aggregation)
Definition
$locfNew in version 5.2.
Last observation carried forward. Sets values for
nulland missing fields in a window to the last non-null value for the field.$locfis only available in the$setWindowFieldsstage.
Syntax
The $locf expression has this syntax:
{ $locf: <expression> } 
For more information on expressions, see Expression Operators.
Behavior
If a field being filled contains both null and non-null values,
$locf sets the null and missing values to the field's
last known non-null value according to the sort order specified in
$setWindowFields.
null and missing field values that appear before non-null values
in the sort order remain null.
If a field being filled contains only null or missing values in a
partition, $locf sets the
field value to null for that partition.
Comparison of $fill and $locf
To fill missing field values based on the last observed value in a sequence, you can use:
The
$fillstage with{ method: "locf" }.When you use the
$fillstage, the field you specify in the output is the same field used as the source data. See Fill Missing Field Values Based on the Last Observed Value.The
$locfoperator inside of a$setWindowFieldsstage.When you use the
$locfoperator, you can set values for a different field than the field used as the source data. See Use Multiple Fill Methods in a Single Stage.
Examples
The examples on this page use a stock collection that contains
tracks a single company's stock price at hourly intervals:
db.stock.insertMany( [    {       time: ISODate("2021-03-08T09:00:00.000Z"),       price: 500    },    {       time: ISODate("2021-03-08T10:00:00.000Z"),    },    {       time: ISODate("2021-03-08T11:00:00.000Z"),       price: 515    },    {       time: ISODate("2021-03-08T12:00:00.000Z")    },    {       time: ISODate("2021-03-08T13:00:00.000Z")    },    {       time: ISODate("2021-03-08T14:00:00.000Z"),       price: 485    } ] ) 
The price field is missing for some of the documents in the
collection.
Fill Missing Values with the Last Observed Value
The following example uses the $locf operator to set missing
fields to the value from the last-observed non-null value:
db.stock.aggregate( [    {       $setWindowFields: {         sortBy: { time: 1 },         output: {            price: { $locf: "$price" }         }       }     } ] ) 
In the example:
sortBy: { time: 1 }sorts the documents in each partition bytimein ascending order (1), so the earliesttimeis first.For documents where the
pricefield is missing, the$locfoperator sets thepriceto the last-observed value in the sequence.
Example output:
[   {     _id: ObjectId("62169b65394d47411658b5f5"),     time: ISODate("2021-03-08T09:00:00.000Z"),     price: 500   },   {     _id: ObjectId("62169b65394d47411658b5f6"),     time: ISODate("2021-03-08T10:00:00.000Z"),     price: 500   },   {     _id: ObjectId("62169b65394d47411658b5f7"),     time: ISODate("2021-03-08T11:00:00.000Z"),     price: 515   },   {     _id: ObjectId("62169b65394d47411658b5f8"),     time: ISODate("2021-03-08T12:00:00.000Z"),     price: 515   },   {     _id: ObjectId("62169b65394d47411658b5f9"),     time: ISODate("2021-03-08T13:00:00.000Z"),     price: 515   },   {     _id: ObjectId("62169b65394d47411658b5fa"),     time: ISODate("2021-03-08T14:00:00.000Z"),     price: 485   } ] 
Use Multiple Fill Methods in a Single Stage
When you use the $setWindowFields stage to fill missing
values, you can set values for a different field than the field you
fill from. As a result, you can use multiple fill methods in a single
$setWindowFields stage and output the results in distinct
fields.
The following pipeline populates missing price fields using
linear interpolation and the last-observation-carried-forward method:
db.stock.aggregate( [    {       $setWindowFields:          {             sortBy: { time: 1 },             output:                {                   linearFillPrice: { $linearFill: "$price" },                   locfPrice: { $locf: "$price" }                }          }    } ] ) 
In the example:
sortBy: { time: 1 }sorts the documents by thetimefield in ascending order, from earliest to latest.output specifies:
linearFillPriceas a target field to be filled.{ $linearFill: "$price" }is the value for thelinearFillPricefield.$linearFillfills missingpricevalues using linear interpolation based on the surroundingpricevalues in the sequence.
locfPriceas a target field to be filled.{ $locf: "$price" }is the value for thelocfPricefield.locfstands for last observation carried forward.$locffills missingpricevalues with the value from the previous document in the sequence.
Example output:
[   {     _id: ObjectId("620ad555394d47411658b5ef"),     time: ISODate("2021-03-08T09:00:00.000Z"),     price: 500,     linearFillPrice: 500,     locfPrice: 500   },   {     _id: ObjectId("620ad555394d47411658b5f0"),     time: ISODate("2021-03-08T10:00:00.000Z"),     linearFillPrice: 507.5,     locfPrice: 500   },   {     _id: ObjectId("620ad555394d47411658b5f1"),     time: ISODate("2021-03-08T11:00:00.000Z"),     price: 515,     linearFillPrice: 515,     locfPrice: 515   },   {     _id: ObjectId("620ad555394d47411658b5f2"),     time: ISODate("2021-03-08T12:00:00.000Z"),     linearFillPrice: 505,     locfPrice: 515   },   {     _id: ObjectId("620ad555394d47411658b5f3"),     time: ISODate("2021-03-08T13:00:00.000Z"),     linearFillPrice: 495,     locfPrice: 515   },   {     _id: ObjectId("620ad555394d47411658b5f4"),     time: ISODate("2021-03-08T14:00:00.000Z"),     price: 485,     linearFillPrice: 485,     locfPrice: 485   } ]