Discussion 3: Loop Invariants
Solutions
double[] array of sensor measurements. We expect all of the readings to be non-negative; however, occasionally the sensor will record a spurious negative value. To "clean" the data, we'd like to "clip" these negative values, replacing them with 0. For example, the array
We'll define a method
cleanData() to carry out this operation.
|
|
|
|
Throughout the method, we’ll use the loop variable i to keep track of the index of the next array entry that we will inspect. Let’s suppose that we want to carry out the cleaning from right to left.
How should we initialize i?
i to the rightmost index in the array i = readings.length - 1.
What should our loop guard be to ensure that we exit the loop immediately after inspecting all of the array entries?
i advances past the leftmost index in the array, or when i < 0. Negating this, we should guard our loop on the condition i >= 0.
cleanData().
Use your answers from parts (a) and (b) to write the definition of the cleanData() method.
Add a comment above the loop documenting the loop invariant. Then, fill in the loop body.
|
|
|
|
ints is a mountain if its entries (weakly) increase to the array's maximum value and then (weakly) decrease after this. More concretely, an array int[] nums is a mountain if there is some index i with 0 <= i < nums.length for which nums[..i] is a non-decreasing sequence and nums[i..] is a non-increasing sequence. For example,
i = 3. However,
isMountain() that determines whether a given int[] array is a mountain.
|
|
|
|
In addition to a loop index variable j, what other variables will you need to keep track of our progress as we traverse the array? (Hint: Our solution uses one additional boolean variable).
boolean variable called goingDown to indicate if we are ascending or descending the mountain.
return conditions: reaching the end of the array and certifying that it is a mountain, versus detecting that the array is not a mountain and returning early.
Using the “Pre” diagram, how should we initialize the loop variables? Briefly explain why your initialization makes the loop invariant true at the start of the loop.
j = 1 and goingDown = false. When we do this, the invariant diagram implies that the non-increasing section is empty. This means that the indexing variable j will be immediately after the non-decreasing section. By initializing j = 1, we are consistent with the precondition diagram, where the first element is in the non-decreasing section. We can verify this since the precondition guarantees us at least an array of length 1, and a 1-element array is trivially non-decreasing.
What should be our loop guard? When we exit from the loop, what will our return value be?
true case, as the false case returns early when identifying an inconsistency in the mountain. To make the invariant look like the true postcondition diagram, we have j = nums.length. Thus, we want to keep on iterating until j = nums.length, meaning the loop should run while j < nums.length (or j != nums.length). After we exit from the loop, we have verified that the array is mountainous, and we can return true. If the array were not mountainous, somewhere through the iteration, the false postcondition would have been satisfied, and the loop would have returned early.
Use your answers from parts (c) and (d) to set up your loop in the isMountain() method. Add a comment above the loop documenting the loop invariant. Then, fill in the loop body.
|
|
|
|
j by 1. We are making progress towards nums.length in each iteration. To analyze how the loop maintains the invariant, we analyze four cases:Case 1: goingDown = false && nums[j] >= nums[j-1]
Since we are ascending the mountain and the current element is at least the previous, nums[j] can be included as part of the non-decreasing section. No extra work is needed besides incrementing j.
Case 2: goingDown = false && nums[j] < nums[j-1]
At this point, we have reached the peak of the (potentially) mountainous array. We were going up but found an element that was less than the previous. This element must be part of the non-increasing section, which must be empty if goingDown = false. To rectify this, we set goingDown = true, which should make logical sense, since we are now descending the mountain.
Case 3: goingDown = true && nums[j] <= nums[j-1]
Much like case 1, this is consistent with the ongoing direction, i.e., continuing to go down the mountain. No extra work is needed besides incrementing j.
Case 4: goingDown = true && nums[j] > nums[j-1]
At this point, we have already hit the peak and are expecting all subsequent elements to be part of the non-increasing section. This violates the mountain property since the array is now going back uphill, so we can return false.
ints into three segments by comparing them to a pivot, another special int value. We'll move all the entries that are less than the pivot to the beginning of the array, followed by all the entries that are equal to the pivot, followed by all the entries that are greater than the pivot. Soon in lecture, we'll see that this type of pivot partitioning forms the basis for the Quicksort sorting algorithm.
|
|
|
|
int loop variables (i, j, and k) and the following loop invariant:
Adjust the “Inv” array diagram back in time to draw the “Pre” array diagram. Use this to determine the correct initializations of the loop variables i, j, and k.
i = 0, j = nums.length, and k = nums.length.
Adjust the “Inv” array diagram forward in time to draw the “Post” array diagram. Use this to determine the condition we should use to guard our loop.
i < k.
Within the while loop of the partition3Way() method, we’ll start by reading the value of nums[i]. For each of the following cases, describe what must be done in the loop body to make progress and restore the loop invariant.
nums[i] < pivot:
i to absorb nums[i] into the "< pivot" range.
nums[i] == pivot:
nums[i] and nums[k-1] and then decrement k.
nums[i] > pivot:
nums[i] and nums[k-1] and then swap nums[k-1] and nums[j-1] before decrementing both j and k.
Use your answers from parts (a), (b), and (c) to write the definition of the partition3Way() method. Be sure to document your loop invariant (using proper range notation) in a multi-line comment above the loop declaration.
|
|
|
|