Dynamic Programming


Imagine you need to travel by taxi from Thiruvananthapuram to Ernakulam.You have several possible routes through different cities, and your goal is to find the one that gets you to Ernakulam in the shortest time, which you call the optimal route. Your cousin in Alappuzha knows the best route from Alappuzha 
to Ernakulam. Given that any optimal route from Thiruvananthapuram must pass through Alappuzha, you can simplify your task by first finding the best route from Thiruvananthapuram to Alappuzha, which is closer and has fewer possible routes. Once you have this route, you can follow the one your cousin has suggested from Alappuzha to Ernakulam.

This approach is based on the principle of optimality, which states that any optimal route from Thiruvananthapuram to Ernakulam via Alappuzha must have optimal sub-routes. Specifically, the segment from Alappuzha to Ernakulam must be the best route between these two cities, and the segment from Thiruvananthapuram to Alappuzha must also be optimal. More generally, if you have an optimal route consisting of cities C1,C2, . . . ,Cp, then each segment of this route (from C1 to C2, C2 to C3, etc.) must be optimal on itsown. By solving the problem in smaller parts—finding the best route fromThiruvananthapuram to Alappuzha and using the known optimal route from Alappuzha to Ernakulam—you can effectively solve the larger problem. This principle, known as Bellman’s principle of optimality, was developed by Richard Bellman in the late 1940s and applies to various optimization problems beyond travel routes. In this section, we will explore how dynamic programming leveragesthis principle to tackle complex optimization challenges.

Dynamic programming (DP) is a method for solving problems by breaking them down into smaller overlapping subproblems, solving each subproblem just once, and storing their solutions. It is particularly useful for optimization problems where the problem can be divided into simpler subproblems that are solvedindependently and combined to form a solution to the original problem.

Dynamic Programming was first introduced by Richard Bellman in the 1950s as part of his research in operations research and control theory. In this context, the term “programming” does not relate to coding but refers to the process of optimizing a series of decisions.

What is Dynamic Programming?

Dynamic Programming (DP) is an algorithmic approach used to solve problems by breaking them down into simpler sub-problems and storing the results of these sub-problems to avoid redundant computations. It is particularly effective for problems that exhibit overlapping sub-problems and optimal substructure.

Key Concepts in Dynamic Programming

Dynamic Programming is built on two key principles:

  1. Overlapping Sub-Problems: In many problems, the same sub-problems are solved multiple times. Instead of solving the same sub-problem repeatedly, DP solves it once and stores the result for future reference. This reduces the overall computation time.

  2. Optimal Substructure: A problem has optimal substructure if an optimal solution to the problem can be constructed from optimal solutions to its sub-problems. This means that the solution to the main problem depends on the solutions to its sub-problems.


Approaches in Dynamic Programming
Dynamic Programming can be implemented using two main approaches: memoization (top-down) and tabulation (bottom-up).

Memoization (Top-Down Approach)
Memoization involves solving the problem recursively and storing the results of subproblems in a table (usually a dictionary or array). This way, each subproblem is solved only once, and subsequent calls to the subproblem are served from the stored results.
Steps:
1. Identify the base cases.
2. Define the recursive relation.
3. Store the results of subproblems in a table.
4. Use the stored results to solve larger subproblems.

#Example: n'th Fibonacci number using memoization
# 1 1 2 3 5 8 13 21
def fib(n, memo={}):
    if n in memo:
        return memo[n]
    if n <= 1:
        return n
    memo[n] = fib(n-1, memo) + fib(n-2, memo)
    return memo[n]
memo={}
print(fib(8,memo))

The function fib takes two arguments:
 n: The position in the Fibonacci sequence for which we want to find the Fibonacci number.
memo: A dictionary used to store previously computed Fibonacci numbers. It defaults to an empty dictionary if not provided.

The fib function leverages memoization to optimize the calculation of Fibonacci numbers by storing the results of previously computed numbers in a dictionary. This approach significantly reduces the time complexity of the algorithm from exponential to linear by avoiding redundant calculations.

Tabulation (Bottom-Up Approach)
Tabulation involves solving the problem iteratively and filling up a table (usually an array) in a bottom-up manner. This approach starts with the smallest subproblems and uses their solutions to construct solutions to larger subproblems.

1. Identify the base cases.
2. Define the table to store solutions to subproblems.
3. Fill the table iteratively using the recursive relation.
4. Extract the solution to the original problem from the table.
Example: Fibonacci sequence using tabulation
def fib(n):
    print("x")
    if n <= 1:
        return n
    dp = [0] * (n + 1)
    dp[1] = 1
    for i in range(2, n + 1):
        dp[i] = dp[i-1] + dp[i-2]
    return dp[n]
print(fib(6))

The given code defines a function fib that calculates the n-th Fibonacci number using a tabular approach. Dynamic programming is a method for solving problems by breaking them down into simpler subproblems and storing the solutions to these subproblems in a table to avoid redundant calculations.

Tabulation tends to be more memory-efficient and can be faster than memoization due to its iterative nature. However, it requires careful planning to setup the data structures and dependencies correctly.

Steps in Dynamic Programming

Dynamic Programming typically involves the following steps:

  1. Identify the sub problem: Break down the problem into smaller subproblems. Determine what the subproblems are and how they can be combined to solve the original problem.

  2. Formulate the Recurrence Relation: Express the solution to the problem in terms of the solutions to smaller subproblems. This usually involves finding a recursive formula that relates the solution of a problem to the solutions of its subproblems.

  3. Base Case: Determine the simplest sub-problem that can be solved directly, usually the smallest possible instance of the problem.

  4. Choose a Memoization or Tabulation Strategy: Decide whether to use a top-down approach with memoization or a bottom-up approach with tabulation.

  • Memoization (Top-Down): Solve the problem recursively and 

    store the results of subproblems in a table (or dictionary) to avoid redundant computations.
  • Tabulation (Bottom-Up): Solve the problem iteratively, starting with the smallest subproblems and building up the solution to the original problem.
        5.Implement the Solution: Using the information stored in the DP table, construct the solution to              the original problem.

       6.

Optimize Space Complexity (if necessary): 
Sometimes, it is possible to optimize space                         complexity by using less memory. For example, if only a few previous states are needed to 
          compute the current state, you can reduce the size of the table.

Advantages of Dynamic Programming

  • Efficiency: By storing the results of sub-problems, DP avoids redundant calculations, making it significantly more efficient than naive recursive approaches, especially for large input sizes.

  • Optimal Solutions: DP guarantees finding the optimal solution for problems with optimal substructure by exploring all possibilities and choosing the best one.

  • Versatility: DP can be applied to a wide range of problems across different domains.

Disadvantages of Dynamic Programming

  • Space Complexity: Storing the results of sub-problems can consume a lot of memory, especially for problems that require large DP tables or multiple dimensions.

  • Complexity in Problem Formulation: Identifying the right state and recurrence relation can be challenging, particularly for more complex problems. The process often requires deep problem analysis and careful planning.

  • Overhead of Table Management: Managing and maintaining the DP table or memoization structure can add overhead to the algorithm.


Dynamic Programming is a versatile and efficient method for solving complex problems that can be broken down into simpler overlapping sub-problems. By storing and reusing the results of these sub-problems, DP dramatically improves the performance of algorithms, making it a vital tool in computer science and optimization. Understanding how to apply DP effectively can open up solutions to a wide range of challenging problems.

Examples:

The Knapsack Problem

The knapsack problem is a classical example of a problem that can be solved using dynamic programming. The problem is defined as follows:

Given weights and values of n items, put these items in a knapsack of capacity W to get the maximum total value in the knapsack. Each item can only be taken once.

Consider the following example:

• Capacity of the knapsack W = 50
• Number of items n = 3
• Weights of the items: w = [10, 20, 30]
• Values of the items: v = [60, 100, 120]

We want to find the maximum value we can carry in the knapsack. For this example, the maximum value we can carry in the knapsack of capacity 50 is 220.

Longest Common Subsequence (LCS):

The Longest Common Subsequence (LCS) problem is a fundamental string comparison challenge that identifies the longest sequence common to two or more strings. Unlike a substring, a subsequence maintains the order of characters but does not need to be contiguous. Brute-force solutions to the LCS problem involve examining all possible subsequences to determine the longest common one, which is computationally expensive and impractical for longer strings due to its exponential time complexity.

Dynamic Programming offers a more efficient approach by dividing the problem into smaller subproblems and using memoization or tabulation to store intermediate results.

Rod Cutting Problem

The Rod Cutting problem is a classic optimization problem, relevant in fields such as manufacturing and finance. Given a rod of length n and a price table for various lengths, the goal is to determine the maximum revenue achievable by cutting the rod into pieces and selling them.

A brute-force approach involves evaluating all possible cutting combinations and calculating the revenue for each, which becomes infeasible for longer rodsdue to its high complexity. Dynamic Programming addresses this issue by breaking the problem into smaller subproblems and using memoization or tabulation to find the optimal solution efficiently.

Comments

Popular posts from this blog

Algorithmic Thinking with Python UCEST 105- KTU First Semester BTech Course 2024 scheme notes pdf - Dr Binu V P 9847390760

Lab Experiments and Solutions - Algorithmic thinking with Python KTU S1 2024 scheme

UCEST 105 Lab Cycle - 1