About 90 minutes
Here are links to lessons that should be completed before this lesson:
Participants will be able to:
Writing memoization into code can be very simple:-
This lesson follows the Memoization Slideshow.
Read through the history of Memoization and ‘What is memoization’ slideshow and get familiar with the new vocabulary.
Before we read about Memoization, we need to understand that it is a part of Dynamic Programming. Dynamic Programming is a method for solving a complex problem by breaking it down into a collection of simpler subproblems, solving each of those subproblems just once, and storing their solutions using a memory-based data structure (array, map, etc). Dynamic programming is mainly an optimization over plain recursion. Wherever we see a recursive solution that has repeated calls for the same inputs, we can optimize it using Dynamic Programming. The idea is to simply store the results of subproblems so that we do not have to re-compute them when needed later. Memoization (top-down approach), along with its “sibling” tabulation (bottom-up approach), are the two main Dynamic Programming techniques.
Memoization is when your code has a function that returns the same results every time given the same input. The result is stored in a cache in order to be used again without needing to take time to re-run the function. Remember that memoization is all about saving time.
Memoization and tabulation are two strategies used in dynamic programming. Memoization solves the top problem and stores the answer to be used later. Memoization is useful when you do not need to solve all the problems to get to the answer. Tabulation solves all problems starting at the bottom and stores the results in a matrix to be called on later.
Take a minute to ask yourself if the bolded vocabulary on the first three slides is known or unknown. Take a minute to understand the new vocabulary by talking to a peer or searching online for examples and definitions.
On slide 5, determine if the functions are pure functions. Discuss and select the functions that are pure functions.
On slide 6 reveal the results. State why the other functions are impure functions.
The Fibonacci Sequence problem is a well-known example of recursion and provides a great way to see many different options for solving the problem and determining the best runtime. The Fibonacci Sequence is created so that each number is the sum of the two preceding numbers.
Pause on slide 9 and write the code to implement the Fibonacci Sequence with a while loop.
View the solution on slide 10 and compare to your own code.
Next, try to implement the Fibonacci Sequence using recursion. Really do this before looking at the solution on slide 12!
Notice the difference in Time Complexity between using recursion and a while loop.
Finally, look at the example code of the Fibonacci Sequence using memoization on slide 13 and again notice the Time Complexity and Space Complexity.
Take a look at the comparison chart showing runtime complexity and space complexity for different sample size. What is your biggest take away from slides 17 and 18?
Use the resources on the next few slides to solidify your understanding. Go back and review any of the vocabularies that you were unsure about at first after you watch the video and go to the links provided.
List things that apprentices might not realize, might assume at first, or should avoid.
Work step by step to change a recursion of Fibonacci Sequence to a memoization version:
function fibonacci(num) {
if (num <= 1) return 1;
return fibonacci(num - 1) + fibonacci(num - 2);
}
How do we need to change the function definition? Change this line to include ‘memo’ as an argument in the function definition: function fibonacci(num)
What would it look like?
Ans: function fibonacci(num, memo)
How do we define ‘memo’ in the function which was not defined when we used recursion? What would you add to the function?
Ans: Create a hash table to store the data to be used again when called upon. memo = memo || {};
When passed an argument, check to see if the result is stored in the memo: if (memo[num]) return memo[num]
Write the rest of the function checking for 0 and then calling the memo if it exists.
if (num <= 1) return 1;
return memo[num] = fibonacci(num - 1, memo) + fibonacci(num - 2, memo);
}
Complete solution using memoization:
function fibonacci(num, memo) {
memo = memo || {};
if (memo[num]) {
return memo[num];
}
if (num <= 1){
return 1;
}
return memo[num] = fibonacci(num - 1, memo) + fibonacci(num - 2, memo);
}
Try this problem from Interview Cake
Explain the benefits of using memoization including the differences in Time Complexity and Space Complexity.