Skip to content

Commit b56bbde

Browse files
authored
Review recursion (#181)
1 parent 43d4819 commit b56bbde

8 files changed

+35
-12
lines changed

recursion/README.md

+5-7
Original file line numberDiff line numberDiff line change
@@ -3,11 +3,11 @@
33
Recursion is a computational technique that implements a [divide-and-conquer](../dnc) approach to problem-solving by breaking down a complex problem into smaller sub-problems. It consists of two components:
44

55
* One or more base cases that provide output for simple inputs and terminate recursion
6-
* A recursive case that combines the outputs obtained from recursive function calls to generate a solution for the original problem.
6+
* A recursive case that combines the outputs obtained from recursive function calls to generate a solution for the original problem
77

88
Although recursions enhance code readability, they are usually inefficient and challenging to debug. Consequently, unless they provide a more efficient solution to a problem, such as in the case of [quicksort](../dnc/quick_sort_test.go), they are generally not preferred.
99

10-
During execution, a program typically stores function variables in a memory area known as the stack before executing recursion. The recursive function may assign different values to the same variables during each recursion. When the recursion ends, the stack pops and remembers the values. However, the stack will grow with each call if recursion continues indefinitely, causing the familiar stack overflow error. Since recursion employs the stack to execute, every recursive problem can be converted into an iterative one. This transformation, however, typically leads to more complex code and may require a [stack](../stack).
10+
During execution, a program typically stores function variables in a memory area known as the stack before executing recursion. The recursive function may assign different values to the same variables that are stored separately for each call during each recursion. When the recursion ends, the stack pops and remembers the values. However, the stack will grow with each call if recursion continues indefinitely, causing the familiar stack overflow error. Since recursion employs the stack to execute, every recursive problem can be converted into an iterative one. This transformation, however, typically leads to more complex code and may require a [stack](../stack).
1111

1212
## Implementation
1313

@@ -16,9 +16,7 @@ The computation of the nth Fibonacci number can be achieved with recursion. For
1616
```Go
1717
package main
1818

19-
import (
20-
"fmt"
21-
)
19+
import "fmt"
2220

2321
func main() {
2422
for i := 1; i <= 10; i++ {
@@ -37,8 +35,8 @@ func fibonacci(n int) int {
3735
When formulating recursive algorithms, it is essential to consider the following four rules of recursion:
3836

3937
1. It is imperative to establish a base case, or else the program will terminate abruptly
40-
2. The algorithm should progress toward the base case at each recursive call.
41-
3. Recursive calls are presumed effective; thus, traversing every recursive call and performing bookkeeping is unnecessary.
38+
2. The algorithm should progress toward the base case at each recursive call
39+
3. Recursive calls are presumed effective; thus, traversing every recursive call and performing bookkeeping is unnecessary
4240
4. Use memoization, a technique that prevents redundant computation by caching previously computed results, can enhance the algorithm's efficiency.
4341

4442
## Complexity

recursion/climbing_stairs_test.go

+13
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,19 @@ TestClimbingStairs tests solution(s) with the following signature and problem de
99
1010
Given n the number of steps, return in how many ways you can climb these stairs if you are
1111
only able to climb 1 or 2 steps at a time.
12+
13+
For example given 5 we can climb the stairs in the following ways:
14+
15+
1, 1, 1, 1, 1
16+
1, 1, 1, 2
17+
1, 1, 2, 1
18+
1, 2, 1, 1
19+
2, 1, 1, 1
20+
2, 2, 1,
21+
1, 2, 2,
22+
2, 1, 2,
23+
24+
So the algorithm should return 8.
1225
*/
1326
func TestClimbingStairs(t *testing.T) {
1427
tests := []struct {

recursion/exponentiation_test.go

+2
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,8 @@ TestPowerOf tests solution(s) with the following signature and problem descripti
88
func PowerOf(x, n int) int
99
1010
Given x and n, return x raised to the power of n in an efficient manner.
11+
12+
For example given x=2 and n=3 the algorithm should return 8.
1113
*/
1214
func TestPowerOf(t *testing.T) {
1315
tests := []struct {

recursion/expression_operators_test.go

+2-2
Original file line numberDiff line numberDiff line change
@@ -7,12 +7,12 @@ TestExpressionOperators tests solution(s) with the following signature and probl
77
88
func ExpressionOperators(list []int, target int) string
99
10-
Given a list of numbers representing operands in an equation, and a target integer representing
10+
Given a slice of numbers representing operands in an equation, and a target integer representing
1111
the result of the equation, return a string representing operators that can be inserted between
1212
the operands to form the equation and yield the target result.
1313
Only + and - operators are allowed and the are assumed to have the same priority
1414
15-
For example given {1,5,3} and 3, return +- because 1+5-3 = 3.
15+
For example given {1,5,3} and 3 return {+,-} because 1+5-3 = 3.
1616
*/
1717
func TestExpressionOperators(t *testing.T) {
1818
tests := []struct {

recursion/is_palindrome_test.go

+3-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,9 @@ TestIsPalindrome tests solution(s) with the following signature and problem desc
77
88
func IsPalindrome(s string) bool
99
10-
Given a string like `abba` return true if it's a palindrome and false otherwise.
10+
Given a string return true if it's a palindrome and false otherwise.
11+
12+
For example given `abba` return true. Given `abca` return false.
1113
*/
1214
func TestIsPalindrome(t *testing.T) {
1315
tests := []struct {

recursion/multiplication_test.go

+2
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,8 @@ TestMultiplication tests solution(s) with the following signature and problem de
99
1010
Given two integers, return their product using recursion and without using the
1111
multiplication operator.
12+
13+
For example given 2 and 3 return 6.
1214
*/
1315
func TestMultiplication(t *testing.T) {
1416
tests := []struct {

recursion/regular_expression_test.go

+4
Original file line numberDiff line numberDiff line change
@@ -13,6 +13,9 @@ Given an input and a regular expression pattern where
1313
`*` denotes to zero or more of the proceeding characters
1414
1515
Write a recursive function to return true if the input matches the pattern and false otherwise.
16+
17+
For example given input `aa` and pattern `a*` the algorithm should return true, but given the same
18+
pattern and "ba" it should return false.
1619
*/
1720
func TestRegularExpressions(t *testing.T) {
1821
tests := []struct {
@@ -27,6 +30,7 @@ func TestRegularExpressions(t *testing.T) {
2730
{"aa", "*", false},
2831
{"aa", "*a", false},
2932
{"aa", "a*", true},
33+
{"ba", "a*", false},
3034
{"aa", ".", false},
3135
{"ab", ".", false},
3236
{"ad", "d", false},

recursion/reverse_number_test.go

+4-2
Original file line numberDiff line numberDiff line change
@@ -7,8 +7,9 @@ TestReverseDigits tests solution(s) with the following signature and problem des
77
88
func ReverseDigits(n int) int
99
10-
Given an integer like 321 return a reversed number using recursion where the same digits
11-
are repeated in the reverse order like 321.
10+
Given an integer reverse the order of the digits.
11+
12+
For example given 123 return 321.
1213
*/
1314
func TestReverseDigits(t *testing.T) {
1415
tests := []struct {
@@ -19,6 +20,7 @@ func TestReverseDigits(t *testing.T) {
1920
{12, 21},
2021
{112, 211},
2122
{110, 11},
23+
{123, 321},
2224
}
2325

2426
for i, test := range tests {

0 commit comments

Comments
 (0)