movingProduct()

Calculate the moving product of a Stream<BigDecimal> looking back windowSize elements.

Implementation Notes

This implementation is suitable for Stream<BigDecimal>, for a version that takes a user-specified mapping function see movingProductBy(). By default, nulls are ignored and play no part in calculations, see treatNullAs() and treatNullAsOne() below for ways to change this behavior. The default MathContext for all calculations is MathContext.DECIMAL64, but this can be overridden (see withMathContext(), below).

Signatures

movingProduct(int windowSize)

  • windowSize - How many trailing elements to calculate the product over

Additional Methods

MethodPurpose
excludePartialValues()When calculating the moving product, and the full size of the window has not yet been reached, the gatherer should suppress emitting values until the lookback window is full. See example.
treatNullAsOne()When encountering a null value in a stream, treat it as BigDecimal.ONE instead. See example.
treatNullAs(BigDecimal replacement)When encountering a null value in a stream, treat it as the given replacement value instead. See example.
withMathContext(MathContext mathContext)Replace the MathContext used for all mathematical operations performed by this gatherer. See example.
withOriginal()Include the original input value from the stream in addition to the calculated value in a WithOriginal record. See example.

Note: treatNullAsZero() is also a valid method on this gatherer, but it only makes sense in a very narrow set of circumstances.

Examples

Moving product of window size 3

Stream
    .of("1.0", "2.0", "10.0", "2.0")
    .map(BigDecimal::new)
    .gather(Gatherers4j.movingProduct(3))
    .toList();

// [ 
//   BigDecimal("1.0"), 
//   BigDecimal("2.00"), 
//   BigDecimal("20.000"), 
//   BigDecimal("40.000") 
// ]

Excluding partial values

Showing that in-process moving product values are not emitted for each element until the lookback window has been filled.

Stream
    .of("1.0", "2.0", "10.0", "2.0")
    .map(BigDecimal::new)
    .gather(Gatherers4j.movingProduct(3).excludePartialValues())
    .toList();

// [ 
//   BigDecimal("20.000"), 
//   BigDecimal("40.000") 
// ]

Showing nulls are ignored by default

Stream
    .of(null, null, new BigDecimal("10.0"), new BigDecimal("2.0"), new BigDecimal("1.0"))
    .gather(Gatherers4j.movingProduct(3))
    .toList();

// [ 
//   BigDecimal("10.0"),
//   BigDecimal("20.00"),
//   BigDecimal("20.000")
// ]

Treating null as one

Stream
    .of(null, null, new BigDecimal("10.0"), new BigDecimal("2.0"))
    .gather(Gatherers4j.movingProduct(3).treatNullAsOne())
    .toList();

// [ 
//   BigDecimal("1"), 
//   BigDecimal("1"), 
//   BigDecimal("10.0"), 
//   BigDecimal("20.00") 
// ]

Replacing null with another BigDecimal

Stream
    .of(null, null, new BigDecimal("10.0"), new BigDecimal("2.0"))
    .gather(Gatherers4j.movingProduct(3).treatNullAs(new BigDecimal("3.5")))
    .toList();

// [ 
//   BigDecimal("3.5"), 
//   BigDecimal("12.25"), 
//   BigDecimal("122.500"), 
//   BigDecimal("70.000") 
// ]

Specifying a new MathContext

Stream
    .of("1.111", "2.222", "10.333", "2.444")
    .map(BigDecimal::new)
    .gather(Gatherers4j
        .movingProduct(3)
        .withMathContext(new MathContext(3, RoundingMode.DOWN))
    )
    .toList();

// [ 
//   BigDecimal("1.11"), 
//   BigDecimal("2.46"), 
//   BigDecimal("25.4"), 
//   BigDecimal("55.7") 
// ]

Emitting a record containing the original and calculated values

Stream
    .of("1.0", "2.0", "10.0", "2.0")
    .map(BigDecimal::new)
    .gather(Gatherers4j.movingProduct(3).withOriginal())
    .toList();

// [
//   WithOriginal[original=1.0, calculated=1.0], 
//   WithOriginal[original=2.0, calculated=2.00], 
//   WithOriginal[original=10.0, calculated=20.000], 
//   WithOriginal[original=2.0, calculated=40.000]
// ]