Find the ways of climb up the stairs

Problem


A child is running up a staircase with n steps, and can hop either 1 step, 2 steps,

or 3 steps at a time. Implement a method to count how many possible ways the

a child can run up the stairs.



Answer

This question is from the "Cracking Code Interview book" and it is the same as the "Finding the number of ways to climb stairs".

Recursion through Depth first search


The simplest way is to recursively perform the DFS with backtracking. Given 3 options to choose from, the running time is O(3N)
, where N is the number of stairs.

As I mentioned earlier, it is not an optimal algorithm.

Here is the complete code

#include<iostream>
using namespace std;
void search(int n, int cur, int& count)
{
if (cur >= n)
{
if (cur == n)
count++;
return;
}
for(int i = 1; i<=3; i++)
{
cur+= i;
search(n, cur, count);
cur-= i;
}
}
int count_runup_stairs(int n)
{
int count = 0;
search(n, 0, count);
return count;
}
int main()
{
int n = 3;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
n =5;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
n = 8;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
}

More efficient algorithm with Dynamic programming


In the previous algorithm, there are many redundant combinations repeated through the DFS search. As seen in  "Finding the number of ways to climb stairs", the ways to climb to Nth stairs can be expressed as below: Cn=Cn1+Cn2+Cn3
, where Cn is the ways to run up to Nth stair. In writing, the ways to run up to  Nth stair is the sum of the ways to run up to N-1th stairs, N-2 th stair, and N-3 stair.

Based on this optimal solution structure we can build the solution bottom up. 

First, we need to set the base cases straight. For the first three stairs.

C0=1
C1=C11+C12+C13=C0=1
C2=C21+C22+C23=C1+C0=2
C3=C31+C32+C33=C2+C1+C0=4

From these base cases, we can start building up the solutions from the 4th stairs following this equation.

Cn=Cn1+Cn2+Cn3

Please, refer to "Finding the number of ways to climb stairs" for a more detailed explanation. The only difference with this question is that "Finding the number of ways to climb stairs" had only one or two stairs choices.

The running time of this algorithm is O(N).

Here is the complete code in C++.

#include<iostream>
using namespace std;
int count_runup_stairs(int n)
{
int c_1 = 4;
int c_2 = 2;
int c_3 = 1;
int cur = 0;
if (n == 1)
return 1;
else if (n== 2)
return 2;
else if (n== 3)
return 4;
for (int i = 4; i <= n; i++)
{
cur = c_1 + c_2 + c_3;
c_3 = c_2;
c_2 = c_1;
c_1 = cur;
}
return cur;
}
int main()
{
int n = 3;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
n =5;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
n = 8;
cout<<" Ways to climb "<<n<<" stairs : "<< count_runup_stairs(n)<<endl;
}
~

Practice statistics

10.58: to write up the inefficient solution
26:00: to write up the second solution

UPDATE(2022-06-14): Solve the problem again after refreshing my memory on the dynamic programming. Took 3 mins to write the code with the critical mistake of setting the wrong value for C0, 0 instead of 1.

# code
def count_ways_to_stairs(num_steps):
# prepopulate the array with 0
c = [0 for i in range(num_steps + 1)]
# set the base cases
c[0] = 1 # important, need to set the base case correctly. It should be 1 even though, there is no case for getting to 0th stair
c[1] = 1
c[2] = 2 # from c 0 and c 1
c[3] = 4
if num_steps <= 3:
return c[num_steps]
for i in range(4, num_steps + 1):
c[i] = c[i - 1] + c[i -2] + c[i - 3]
return c[num_steps]
# test
target = 4
result = count_ways_to_stairs(target)
print ("number of ways to get to {}th step = {}".format(target, result))
target = 5
result = count_ways_to_stairs(target)
print ("number of ways to get to {}th step = {}".format(target, result))
target = 6
result = count_ways_to_stairs(target)
print ("number of ways to get to {}th step = {}".format(target, result))
target = 10
result = count_ways_to_stairs(target)
print ("number of ways to get to {}th step = {}".format(target, result))

Comments

Popular posts from this blog

Planting flowers with no adjacent flower plots

Find the shorted path from the vertex 0 for given list of vertices.