Given a N x N matrix where value at cell (i, j) is the cost of moving from a cell (i, j) to cell (i – 1, j – 1), (i – 1, j) or (i, j – 1). Your task is to find the maximum cost path from (N – 1, N – 1) cell to (0, 0) cell in the N x N matrix (0 – based indexing). However, you have some restrictions on the movement from one cell to the other cell. If you are at (i, j) cell and (i + j) is a power of 2, you can only move to (i – 1, j – 1) cell. If (i + j) is not a power of 2 then you can move to (i – 1, j) or (i, j – 1)
Examples:
Input :[1 2 3 1
        4 5 6 1
        7 8 9 1
        1 1 1 1]
Output: 16 
The maximum cost path is: 
(3, 3) -> (3, 2) -> (2, 2) -> (1, 1) -> (0, 0).
Cost pathwise is: 
1 + 1 + 9 + 5 = 16.
Input: [1 2 
        3 4]
Output: 4 
Optimal Substructure:
The problem is a variation of Min-Cost problem. The path to reach (0, 0) from (n-1, n-1) must be through the three cells (i, j-1) or (i-1, j) or (i-1, j-1). A top-down recursive function will be called, for every value of m and n, check if (m+n) is a power of 2 or not. If it is a power of 2, then move to cell(m-1, n-1) and add the value at a[m][n]. Hence the cost will be:
cost = a[m][n] + maxCost(a, m – 1, n – 1)
If it is not a power of 2, then we can move to two of cells (m-1, n) and (m, n-1). So the cost will be:
cost = a[m][n] + max(maxCost(a, m – 1, n), maxCost(a, m, n – 1))
Below is the recursive implementation of the above approach:
C++
| // C++ program for// SP - Wolfish#include <bits/stdc++.h>usingnamespacestd;constintsize = 1000;// Function to find the maxCost of path from// (n-1, n-1) to (0, 0) | recursive approachintmaxCost(inta[][size], intm, intn){    // base condition    if(n < 0 || m < 0)        return-1e9;    // reaches the point    elseif(m == 0 && n == 0)        return0;    else{        // i + j        intnum = m + n;        // check if it is a power of 2,        // then only move diagonally        if((num & (num - 1)) == 0)            returna[m][n] + maxCost(a, m - 1, n - 1);        // if not a power of 2        // then move side-wise        else            returna[m][n] + max(maxCost(a, m - 1, n),                                 maxCost(a, m, n - 1));    }}// Function to return the maximum costintanswer(inta[][size], intn){    // calling dp function to get the answer    returnmaxCost(a, n - 1, n - 1);}// Driver Codeintmain(){    inta[][size] = { { 1, 2, 3, 1 },                      { 4, 5, 6, 1 },                      { 7, 8, 9, 1 },                      { 1, 1, 1, 1 } };    intn = 4;    // Function calling to get the answer    cout << answer(a, n);    return0;} | 
Java
| // Java program for SP - WolfishclassGFG {    staticintsize = 1000;    // Function to find the maxCost of path from    // (n-1, n-1) to (0, 0) | recursive approach    publicstaticintmaxCost(int[][] a, intm, intn)    {        // base condition        if(n < 0|| m < 0) {            return-1;        }        // reaches the point        elseif(m == 0&& n == 0) {            return0;        }        else{            // i + j            intnum = m + n;            // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0) {                returna[m][n] + maxCost(a, m - 1, n - 1);            }            // if not a power of 2            // then move side-wise            else{                returna[m][n] + Math.max(maxCost(a, m - 1, n), maxCost(a, m, n - 1));            }        }    }    // Function to return the maximum cost    publicstaticintanswer(int[][] a, intn)    {        // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1);    }    // Driver Code    publicstaticvoidmain(String[] args)    {        int[][] a = newint[][] { { 1, 2, 3, 1},                                  { 4, 5, 6, 1},                                  { 7, 8, 9, 1},                                  { 1, 1, 1, 1} };        intn = 4;        System.out.println(answer(a, n));    }}/* This code contributed by PrinciRaj1992 */ | 
Python3
| # Python program for# SP - Wolfishsize =1000# Function to find the maxCost of path from# (n-1, n-1) to (0, 0) | recursive approachdefmaxCost(a: list, m: int, n: int) -> int:    # base condition    ifn < 0orm < 0:        returnint(-1e9)    # reaches the point    elifm ==0andn ==0:        return0    else:        # i + j        num =m +n        # check if it is a power of 2,        # then only move diagonally        if(num & (num -1)) ==0:            returna[m][n] +maxCost(a, m -1, n -1)        # if not a power of 2        # then move side-wise        else:            returna[m][n] +max(maxCost(a, m -1, n), maxCost(a, m, n -1))# Function to return the maximum costdefanswer(a: list, n: int) -> int:    # calling dp function to get the answer    returnmaxCost(a, n -1, n -1)# Driver Codeif__name__ =="__main__":    a =[[1, 2, 3, 1],        [4, 5, 6, 1],        [7, 8, 9, 1],        [1, 1, 1, 1]]    n =4    # Function calling to get the answer    print(answer(a, n))# This code is contributed by# sanjeev2552 | 
C#
| // C# program for// SP - WolfishusingSystem;classGFG {    staticintsize = 1000;    // Function to find the maxCost of path from    // (n-1, n-1) to (0, 0) | recursive approach    publicstaticintmaxCost(int[, ] a, intm, intn)    {        // base condition        if(n < 0 || m < 0)            return-1;        // reaches the point        elseif(m == 0 && n == 0)            return0;        else{            // i + j            intnum = m + n;            // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0)                returna[m, n] + maxCost(a, m - 1, n - 1);            // if not a power of 2            // then move side-wise            else                returna[m, n] + Math.Max(maxCost(a, m - 1, n), maxCost(a, m, n - 1));        }    }    // Function to return the maximum cost    publicstaticintanswer(int[, ] a, intn)    {        // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1);    }    // Driver Code    staticvoidMain()    {        int[, ] a = newint[, ] { { 1, 2, 3, 1 },                                  { 4, 5, 6, 1 },                                  { 7, 8, 9, 1 },                                  { 1, 1, 1, 1 } };        intn = 4;        // Function calling to get the answer        Console.Write(answer(a, n));    }    // This code is contributed by DrRoot_} | 
Javascript
| <script>// Javascript program for SP - Wolfishlet size = 1000;// Function to find the maxCost of path from    // (n-1, n-1) to (0, 0) | recursive approachfunctionmaxCost(a,m,n){    // base condition        if(n < 0 || m < 0) {            return-1;        }          // reaches the point        elseif(m == 0 && n == 0) {            return0;        }        else{              // i + j            let num = m + n;              // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0) {                returna[m][n] + maxCost(a, m - 1, n - 1);            }              // if not a power of 2            // then move side-wise            else{                returna[m][n] + Math.max(maxCost(a, m - 1, n), maxCost(a, m, n - 1));            }        }}// Function to return the maximum costfunctionanswer(a,n){    // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1);}// Driver Codelet a=[[ 1, 2, 3, 1 ],[ 4, 5, 6, 1 ],                                  [ 7, 8, 9, 1 ],                                  [ 1, 1, 1, 1 ]];let  n = 4;document.write(answer(a, n));// This code is contributed by rag2127</script> | 
16
Time Complexity: O(2N)
Approach using Memoization
In the above recursion, many sub-problems are being repeatedly called. To reduce the number of repetitive calls, memoization has been used. The common point of observation is that only two parameters value are changing at every function call. So if we memoize the returned value in a dp[][] array, the number of calls will be reduced to N^2. Hence store the computed value of every maxCost(m, n) in dp[m][n]. If the maxCost(m, n) is called more than once, then the extra calls of the function will be reduced by returning the value stored at dp[m][n].
Below is the efficient implementation of the above approach:
C++
| // C++ program for SP - Wolfish#include <bits/stdc++.h>usingnamespacestd;constintsize = 1000;// Function to find the maxCost of path from// (n-1, n-1) to (0, 0)intmaxCost(inta[][size], intm, intn, intdp[][size]){    // base condition    if(n < 0 || m < 0)        return-1e9;    // reaches the point    elseif(m == 0 && n == 0)        return0;    // if the state has been visited previously    elseif(dp[m][n] != -1)        returndp[m][n];    else{        // i + j        intnum = m + n;        // check if it is a power of 2,        // then only move diagonally        if((num & (num - 1)) == 0)            returndp[m][n] = a[m][n] + maxCost(a, m - 1, n - 1, dp);        // if not a power of 2        // then move side-wise        else            returndp[m][n] = (a[m][n] + max(maxCost(a, m - 1, n, dp),                                             maxCost(a, m, n - 1, dp)));    }}// Function to return the maximum costintanswer(inta[][size], intn){    intdp[size][size];    memset(dp, -1, sizeofdp);    // calling dp function to get the answer    returnmaxCost(a, n - 1, n - 1, dp);}// Driver Codeintmain(){    inta[][size] = { { 1, 2, 3, 1 },                      { 4, 5, 6, 1 },                      { 7, 8, 9, 1 },                      { 1, 1, 1, 1 } };    intn = 4;    // Function calling to get the answer    cout << answer(a, n);    return0;} | 
Java
| // Java program for SP - WolfishclassGFG {    staticintsize = 1000;    // Function to find the maxCost of path from    // (n-1, n-1) to (0, 0)    staticintmaxCost(inta[][], intm, intn, intdp[][])    {        // base condition        if(n < 0|| m < 0)            return(int)-1e9;        // reaches the point        elseif(m == 0&& n == 0)            return0;        // if the state has been visited previously        elseif(dp[m][n] != -1)            returndp[m][n];        else{            // i + j            intnum = m + n;            // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0)                returndp[m][n] = a[m][n] + maxCost(a, m - 1, n - 1, dp);            // if not a power of 2            // then move side-wise            else                returndp[m][n] = (a[m][n] + Math.max(maxCost(a, m - 1, n, dp),                                                      maxCost(a, m, n - 1, dp)));        }    }    // Function to return the maximum cost    staticintanswer(inta[][], intn)    {        intdp[][] = newint[size][size];        for(inti = 0; i < size; i++) {            for(intj = 0; j < size; j++) {                dp[i][j] = -1;            }        }        // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1, dp);    }    // Driver Code    publicstaticvoidmain(String[] args)    {        inta[][] = { { 1, 2, 3, 1},                      { 4, 5, 6, 1},                      { 7, 8, 9, 1},                      { 1, 1, 1, 1} };        intn = 4;        // Function calling to get the answer        System.out.println(answer(a, n));    }}// This code has been contributed by 29AjayKumar | 
Python3
| # Python program for SP - Wolfishsize =1000;# Function to find the maxCost of path from# (n-1, n-1) to (0, 0)defmaxCost(a, m, n, dp):    # base condition    if(n < 0orm < 0):        returnint(-1e9);    # reaches the point    elif(m ==0andn ==0):        return0;    # if the state has been visited previously    elif(dp[m][n] !=-1):        returndp[m][n];    else:        # i + j        num =m +n;        # check if it is a power of 2,        # then only move diagonally        if((num & (num -1)) ==0):            dp[m][n] =a[m][n] +maxCost(a, m -1, n -1, dp);            returndp[m][n];        # if not a power of 2        # then move side-wise        else:            dp[m][n] =(a[m][n] +max(maxCost(a, m -1, n, dp), maxCost(a, m, n -1, dp)));            returndp[m][n];    # Function to return the maximum costdefanswer(a, n):    dp =[[0fori inrange(size)] forj inrange(size)] ;    fori inrange(size):        forj inrange(size):            dp[i][j] =-1;                # calling dp function to get the answer    returnmaxCost(a, n -1, n -1, dp);# Driver Codeif__name__ =='__main__':    a =[[ 1, 2, 3, 1],[ 4, 5, 6, 1],[ 7, 8, 9, 1],[ 1, 1, 1, 1]];    n =4;    # Function calling to get the answer    print(answer(a, n));    # This code contributed by Rajput-Ji | 
C#
| // C# program for SP - WolfishusingSystem;     classGFG {    staticintsize = 1000;    // Function to find the maxCost of path from    // (n-1, n-1) to (0, 0)    staticintmaxCost(int[,]a, intm, intn, int[,]dp)    {        // base condition        if(n < 0 || m < 0)            return(int)-1e9;        // reaches the point        elseif(m == 0 && n == 0)            return0;        // if the state has been visited previously        elseif(dp[m, n] != -1)            returndp[m,n];        else{            // i + j            intnum = m + n;            // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0)                returndp[m, n] = a[m, n] + maxCost(a, m - 1, n - 1, dp);            // if not a power of 2            // then move side-wise            else                returndp[m,n] = (a[m,n] + Math.Max(maxCost(a, m - 1, n, dp),                                                    maxCost(a, m, n - 1, dp)));        }    }    // Function to return the maximum cost    staticintanswer(int[,]a, intn)    {        int[,]dp = newint[size,size];        for(inti = 0; i < size; i++)        {            for(intj = 0; j < size; j++)             {                dp[i, j] = -1;            }        }        // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1, dp);    }    // Driver Code    publicstaticvoidMain(String[] args)    {        int[,]a = { { 1, 2, 3, 1 },                    { 4, 5, 6, 1 },                    { 7, 8, 9, 1 },                    { 1, 1, 1, 1 } };        intn = 4;        // Function calling to get the answer        Console.WriteLine(answer(a, n));    }}// This code contributed by Rajput-Ji | 
Javascript
| <script>// Javascript program for SP - Wolfishlet size = 1000;// Function to find the maxCost of path from    // (n-1, n-1) to (0, 0)functionmaxCost(a,m,n,dp){    // base condition        if(n < 0 || m < 0)            return-1e9;         // reaches the point        elseif(m == 0 && n == 0)            return0;         // if the state has been visited previously        elseif(dp[m][n] != -1)            returndp[m][n];        else{             // i + j            let num = m + n;             // check if it is a power of 2,            // then only move diagonally            if((num & (num - 1)) == 0)                returndp[m][n] = a[m][n] + maxCost(a, m - 1, n - 1, dp);             // if not a power of 2            // then move side-wise            else                returndp[m][n] = (a[m][n] + Math.max(maxCost(a, m - 1, n, dp),                                                      maxCost(a, m, n - 1, dp)));        }}// Function to return the maximum costfunctionanswer(a,n){    let dp = newArray(size);        for(let i = 0; i < size; i++) {            dp[i]=newArray(size);            for(let j = 0; j < size; j++) {                dp[i][j] = -1;            }        }         // calling dp function to get the answer        returnmaxCost(a, n - 1, n - 1, dp);}// Driver Codelet a=[[ 1, 2, 3, 1 ],                      [ 4, 5, 6, 1 ],                      [ 7, 8, 9, 1 ],                      [ 1, 1, 1, 1 ]];let n = 4;// Function calling to get the answerdocument.write(answer(a, n));// This code is contributed by avanitrachhadiya2155</script> | 
16
Complexity Analysis:
- Time Complexity: O(N2)
- Auxiliary Space: O(N2)
Note: To implement a bottom-up approach, we need to check if ((m+1) + (n+1)) is a power of 2 or not instead of (m+n) as the moves are in top-down order.
Ready to dive in? Explore our Free Demo Content and join our DSA course, trusted by over 100,000 neveropen!

 
                                    







