346. Moving Average from Data Stream - Detailed Explanation
Problem Statement
Implement a class MovingAverage
that calculates the moving average of the last size
elements from a stream of integers. The class should support two operations:
- MovingAverage(int size): Constructor to initialize the object with the window size.
- double next(int val): Returns the moving average of the last
size
values of the data stream (including the new valueval
).
Example Test Cases
Example 1:
MovingAverage obj = new MovingAverage(3); obj.next(1); // returns 1.0 obj.next(10); // returns (1 + 10) / 2 = 5.5 obj.next(3); // returns (1 + 10 + 3) / 3 = 4.67 obj.next(5); // returns (10 + 3 + 5) / 3 = 6.0
Example 2:
MovingAverage obj = new MovingAverage(2); obj.next(5); // returns 5.0 obj.next(15); // returns (5 + 15) / 2 = 10.0 obj.next(10); // returns (15 + 10) / 2 = 12.5
Hints to Solve
- Use a data structure to store the last
size
values and allow easy addition/removal of elements. - Maintain a running sum so you can compute the new average quickly without summing all elements from scratch each time.
Approaches
Brute Force Approach (Using an Array/List)
- Keep a list of all values seen so far.
- On each call to
next(val)
, add the new value to the list and then compute the sum of the lastsize
elements in the list. - Return the average as this sum divided by
size
(or by the number of elements if fewer thansize
have been seen so far). - Time Complexity: O(size) per query (each
next
call requires iterating over up tosize
elements for the sum).
Optimized Approach (Using a Queue – Sliding Window)
- Use a queue (or deque) to store the most recent
size
values. This naturally discards the oldest value when the size is exceeded. - Maintain a running sum of the values in the queue (the current window sum).
- On each
next(val)
call:- If the queue is already at the maximum size, remove (
popleft()
/poll()
) the oldest value and subtract it from the running sum. - Add the new value to the queue and add it to the running sum.
- Compute the average as
running_sum / current_window_length
.
- If the queue is already at the maximum size, remove (
- Time Complexity: O(1) per query (each
next
operation does a constant amount of work, regardless of the window size).
Code Implementations
Python Solution (Queue Approach)
Java Solution (Queue Approach)
Complexity Analysis
-
Brute Force Approach: Each call to
next
is O(n) where n =size
of the window. This can be inefficient whensize
is large, because computing the sum of the window on every call takes linear time. -
Optimized Queue Approach: Each call to
next
is O(1). We only do a constant amount of work (one addition and possibly one removal), regardless of the window size. This is optimal for this problem.
Common Mistakes
-
Not removing the oldest element: Forgetting to drop the oldest value when the window exceeds the specified size will lead to incorrect averages (and an ever-growing window).
-
Handling the first few elements: Not considering the case when fewer than
size
elements have been seen. In those cases, the average should be taken over all elements that are present so far (the window hasn't fully filled yet).
Edge Cases
- Window Size 1: Every new value should simply be returned as the average (since the window contains only that element).
- Extreme Values: The values can be as low as -10^4 or as high as 10^4. Ensure that the sum (which could be as large as
size * 10^4
) fits in the chosen data type without overflow (in Python it's fine; in Java, anint
might overflow if size is large, so using adouble
or long for the sum is safer).
Alternative Variations
-
Weighted Moving Average: Instead of an unweighted average, each element could have a weight (for example, more recent elements weighted more heavily than older ones).
-
Missing Values: How to handle cases where the data stream might have missing values or nulls—perhaps skipping them or treating them as zeros, depending on the specification.
Related Problems
- Sliding Window Maximum – Find the maximum in each sliding window of size k in an array.
- Design Hit Counter – Design a system that counts hits received in the past 5 minutes using a sliding window approach.
- Running Sum of 1d Array – Although not a sliding window, it involves maintaining a running sum, which is a similar concept.
GET YOUR FREE
Coding Questions Catalog
