Skill

ডাইনামিক প্রোগ্রামিং (Dynamic Programming)

জাভা দিয়ে ডাটা স্ট্রাকচার এবং অ্যালগরিদম (DSA using Java) - Java Technologies

443

ডাইনামিক প্রোগ্রামিং (Dynamic Programming - DP) হল একটি শক্তিশালী অ্যালগরিদমিক কৌশল, যা সাধারণত অপটিমাইজেশন সমস্যাগুলির সমাধান করতে ব্যবহৃত হয়। ডাইনামিক প্রোগ্রামিং মূলত subproblems (ছোট সমস্যা) সমাধান করে এবং তাদের সমাধানগুলি মেমোরিতে সংরক্ষণ করে রাখে, যাতে পরবর্তীতে একই সমস্যা আবার সমাধান করতে না হয়। এটি অনেক ক্ষেত্রেই রিকার্সন এর মতো কাজ করে, তবে এটি মেমোইজেশন বা টেবুলেশন ব্যবহার করে সমস্যাগুলিকে আরো দক্ষভাবে সমাধান করতে সক্ষম হয়।

ডাইনামিক প্রোগ্রামিং এর মূল ধারণা

  1. অপটিমাল সাব-স্ট্রাকচার (Optimal Substructure): একটি বড় সমস্যা (যেমন, সম্পূর্ণ সমস্যার সমাধান) তার ছোট উপ-সমস্যাগুলির সমাধানের মাধ্যমে সমাধান করা যায়।
  2. ওভারল্যাপিং সাব-প্রব্লেমস (Overlapping Subproblems): ডাইনামিক প্রোগ্রামিং কৌশলটি তখনই কার্যকর, যখন একই সাব-সমস্যা বারবার সমাধান করতে হয়। এখানে মেমোইজেশন বা টেবুলেশন ব্যবহার করা হয়।

1. ডাইনামিক প্রোগ্রামিং এর দুটি প্রধান কৌশল

মেমোইজেশন (Memoization):

  • এটি একটি টপ-ডাউন অ্যাপ্রোচ, যেখানে রিকার্সন ব্যবহার করা হয় এবং প্রতিটি সাব-সমস্যার ফলাফল ক্যাশে রাখা হয়, যাতে পুনরায় সেই সাব-সমস্যাটি সমাধান করতে না হয়।
  • মেমোইজেশন সাধারণত রিকার্সিভ অ্যালগরিদমের সাহায্যে কাজ করে।

টেবুলেশন (Tabulation):

  • এটি একটি বটম-আপ অ্যাপ্রোচ, যেখানে ছোট ছোট সাব-সমস্যাগুলির সমাধান শুরু করা হয় এবং টেবিল বা এরে ব্যবহার করে তার ফলাফলগুলি সংরক্ষণ করা হয়।
  • টেবুলেশন সাধারণত iterative পদ্ধতির সাহায্যে কাজ করে এবং এটি রিকার্সন ব্যবহার না করে সরাসরি সমস্যার সমাধান করে।

2. ডাইনামিক প্রোগ্রামিং এর উদাহরণ

ফিবোনাচ্চি সিরিজ (Fibonacci Series)

ফিবোনাচ্চি সিরিজ একটি ক্লাসিক উদাহরণ যেখানে ডাইনামিক প্রোগ্রামিং ব্যবহার করা হয়। ফিবোনাচ্চি সিরিজের প্রথম দুটি মান হল 0 এবং 1, এবং পরবর্তী মান হল পূর্ববর্তী দুই মানের যোগফল। সাধারণভাবে, এটা রিকার্সন এর মাধ্যমে করা হয়, তবে অনেক বার একই সাব-সমস্যা সমাধান করতে হয়, তাই ডাইনামিক প্রোগ্রামিং এর মাধ্যমে এটি দক্ষভাবে সমাধান করা যায়।

1. মেমোইজেশন (Memoization) ব্যবহার করে ফিবোনাচ্চি সিরিজ:

import java.util.HashMap;

public class Fibonacci {

    // Memoization technique: Storing computed Fibonacci numbers
    private static HashMap<Integer, Integer> memo = new HashMap<>();

    // Fibonacci function using Memoization
    public static int fibonacci(int n) {
        if (n <= 1) {
            return n;
        }

        // Check if result already exists in memo
        if (memo.containsKey(n)) {
            return memo.get(n);
        }

        // Otherwise, compute and store the result in memo
        int result = fibonacci(n - 1) + fibonacci(n - 2);
        memo.put(n, result);
        return result;
    }

    public static void main(String[] args) {
        int n = 10; // Finding Fibonacci number for 10
        System.out.println("Fibonacci number at position " + n + " is: " + fibonacci(n));
    }
}

ব্যাখ্যা:

  • এখানে আমরা একটি HashMap ব্যবহার করেছি যেখানে আমরা ফিবোনাচ্চি সিরিজের আগের ফলাফল সংরক্ষণ করি, যাতে একই ফলাফল পুনরায় হিসাব করতে না হয়।
  • ফাংশনটি Memoization কৌশল ব্যবহার করে, অর্থাৎ, যদি কোনো সংখ্যা আগেই হিসাব করা থাকে, তবে সেটি সরাসরি memo থেকে ফেরত দেওয়া হয়।

আউটপুট:

Fibonacci number at position 10 is: 55

2. টেবুলেশন (Tabulation) ব্যবহার করে ফিবোনাচ্চি সিরিজ:

public class FibonacciTabulation {

    // Fibonacci function using Tabulation
    public static int fibonacci(int n) {
        if (n <= 1) {
            return n;
        }

        // Create a table to store results of subproblems
        int[] table = new int[n + 1];
        table[0] = 0;
        table[1] = 1;

        // Fill the table iteratively
        for (int i = 2; i <= n; i++) {
            table[i] = table[i - 1] + table[i - 2];
        }

        return table[n];
    }

    public static void main(String[] args) {
        int n = 10; // Finding Fibonacci number for 10
        System.out.println("Fibonacci number at position " + n + " is: " + fibonacci(n));
    }
}

ব্যাখ্যা:

  • এখানে Tabulation কৌশল ব্যবহার করা হয়েছে, যেখানে একটি array তৈরি করে, প্রথম দুটি ফিবোনাচ্চি সংখ্যা 0 এবং 1 দেয়া হয়।
  • তারপর for loop ব্যবহার করে, প্রতিটি পদকে পূর্ববর্তী দুই পদ যোগ করে হিসাব করা হয়।

আউটপুট:

Fibonacci number at position 10 is: 55

3. ডাইনামিক প্রোগ্রামিং ব্যবহার করার সুবিধা

  • অপটিমাইজেশন: ডাইনামিক প্রোগ্রামিং সমাধানে অপটিমাইজেশনের জন্য ছোট উপ-সমস্যাগুলির ফলাফল সংরক্ষণ করে কাজ করে।
  • কার্যক্ষমতা বৃদ্ধি: পুনরাবৃত্তি কমানোর জন্য একবারে প্রতিটি সাব-সমস্যার সমাধান সংরক্ষণ করা হয়, যা গণনা তাড়াতাড়ি করে।
  • কাজের পুনরাবৃত্তি এড়ানো: এটি পুনরাবৃত্তি করা সাব-সমস্যাগুলির সমাধান সংরক্ষণ করে এবং একে একে কাজ করে।

4. ডাইনামিক প্রোগ্রামিং এর অন্যান্য উদাহরণ

ক Knapsack সমস্যা (0/1 Knapsack Problem)

এই সমস্যায় একটি ব্যাগের সর্বোচ্চ ধারণক্ষমতা দেওয়া হয় এবং কিছু আইটেমের ওজন এবং মূল্য দেওয়া হয়। আপনাকে সর্বোচ্চ মূল্য অর্জন করার জন্য কী আইটেমগুলি ব্যাগে রাখতে হবে তা নির্ধারণ করতে হবে।

Longest Common Subsequence (LCS)

এটি দুটি স্ট্রিংয়ের মধ্যে সর্বাধিক সাধারণ উপসেকোয়েন্স খুঁজে বের করার একটি সমস্যা। এখানে টেবুলেশন এবং মেমোইজেশন উভয় কৌশলই ব্যবহার করা হয়।


5. সারাংশ

ডাইনামিক প্রোগ্রামিং (DP) একটি শক্তিশালী টেকনিক যা অপটিমাইজেশন সমস্যা সমাধান করার জন্য ব্যবহৃত হয়। এটি মেমোইজেশন (top-down approach) এবং টেবুলেশন (bottom-up approach) এর মাধ্যমে কাজ করতে পারে। সাধারণ সমস্যাগুলির মধ্যে ফিবোনাচ্চি সিরিজ, 0/1 Knapsack, এবং Longest Common Subsequence প্রভৃতি রয়েছে, যেখানে ডাইনামিক প্রোগ্রামিং অ্যালগরিদম ব্যবহার করা হয়।

Java তে ডাইনামিক প্রোগ্রামিং সমস্যা সমাধান করতে, টেবুলেশন এবং মেমোইজেশন কৌশলগুলি কার্যকরীভাবে ব্যবহার করা যায়, যা কোডের কার্যক্ষমতা এবং দক্ষতা বৃদ্ধি করে।

Content added By

Dynamic Programming (ডাইনামিক প্রোগ্রামিং) হল একটি শক্তিশালী কৌশল যা পুনরাবৃত্তিক সমস্যাগুলির সমাধান দ্রুত করতে সাহায্য করে। এটি মূলত এক ধরনের "বিভাজন এবং বিজয়" (divide and conquer) পদ্ধতির উন্নত সংস্করণ। ডাইনামিক প্রোগ্রামিং মূলত সমস্যা সমাধান করতে হলে পুনরাবৃত্তি (recursion) ব্যবহার করা হয়, তবে পুনরাবৃত্তি ছাড়াও সাব-প্রোঅব্লেমের সমাধান পুনরায় ব্যবহার করা হয়, যাতে একে একে সমাধান না করতে হয়। এটি সাধারণত বড় সমস্যা ছোট ছোট সাব-প্রোঅব্লেমে ভেঙে সমাধান করা হয়।


Dynamic Programming এর বেসিক কনসেপ্ট

ডাইনামিক প্রোগ্রামিং দুটি প্রধান কনসেপ্টে কাজ করে:

1. Overlapping Subproblems

ডাইনামিক প্রোগ্রামিং একটি সমস্যার পুনরাবৃত্তি (recursion) সমাধানের জন্য ব্যবহৃত হয় যেখানে সমাধান করা সাব-প্রোঅব্লেমগুলি একাধিকবার পুনরায় আসতে পারে। এর মানে হলো, একই সাব-প্রোঅব্লেম বারবার গণনা করতে হয়। ডাইনামিক প্রোগ্রামিং এই পুনরাবৃত্তি সাব-প্রোঅব্লেমগুলির ফলাফল সংরক্ষণ করে, যাতে একই সমস্যার সমাধান একাধিকবার না করতে হয়।

2. Optimal Substructure

ডাইনামিক প্রোগ্রামিং ব্যবহারযোগ্য হয় যখন একটি বড় সমস্যা ছোট ছোট সাব-প্রোঅব্লেমের সমাধান থেকে মিলে। অর্থাৎ, যদি একটি সমস্যা ভালোভাবে বিভক্ত করা যায় এবং প্রতিটি সাব-প্রোঅব্লেমের সমাধানকে একত্রিত করলে মূল সমস্যা সমাধান হয়, তবে ডাইনামিক প্রোগ্রামিং কার্যকর।


Dynamic Programming এর কাজের ধাপ

ডাইনামিক প্রোগ্রামিং সাধারণত দুটি প্রধান পদ্ধতিতে কাজ করে:

  1. Top-Down Approach (Memoization): এই পদ্ধতিতে পুনরাবৃত্তির মাধ্যমে মূল সমস্যা সমাধান করা হয় এবং প্রতি সাব-প্রোঅব্লেমের ফলাফল সংরক্ষণ (memoization) করা হয়, যাতে পরবর্তীতে সেই ফলাফল আবার গণনা না করতে হয়।
  2. Bottom-Up Approach (Tabulation): এই পদ্ধতিতে ছোট ছোট সাব-প্রোঅব্লেমের সমাধান করা হয় এবং তাদের সমাধান ব্যবহার করে বড় সমস্যার সমাধান বের করা হয়। এতে কোন রিকার্সন ব্যবহার হয় না, তবে টেবিল বা অ্যারে ব্যবহার করে সমাধান তৈরি করা হয়।

Dynamic Programming এর উদাহরণ

Fibonacci Sequence (ফিবোনাচ্চি সিকোয়েন্স)

ফিবোনাচ্চি সিকোয়েন্স একটি ক্লাসিক উদাহরণ যেখানে প্রতিটি সংখ্যার মান তার পূর্ববর্তী দুইটি সংখ্যার যোগফল হয়:
F(n) = F(n-1) + F(n-2)

1. Top-Down Approach (Memoization)

import java.util.*;

public class Fibonacci {

    private static Map<Integer, Integer> memo = new HashMap<>();

    public static int fibonacci(int n) {
        if (n <= 1) {
            return n;
        }

        if (memo.containsKey(n)) {
            return memo.get(n); // If result is already computed, return from memo
        }

        int result = fibonacci(n - 1) + fibonacci(n - 2); // Recursively calculate
        memo.put(n, result); // Store the result in memoization table

        return result;
    }

    public static void main(String[] args) {
        int n = 10;
        System.out.println("Fibonacci of " + n + " is: " + fibonacci(n));
    }
}

এখানে fibonacci() ফাংশনটি প্রতিটি ফিবোনাচ্চি সংখ্যার মান বের করার সময় memoization ব্যবহার করে পূর্ববর্তী ফলাফল সংরক্ষণ করছে, যাতে পুনরায় গণনা করতে না হয়।

2. Bottom-Up Approach (Tabulation)

public class Fibonacci {

    public static int fibonacci(int n) {
        if (n <= 1) {
            return n;
        }

        int[] dp = new int[n + 1];
        dp[0] = 0; // Base case
        dp[1] = 1; // Base case

        for (int i = 2; i <= n; i++) {
            dp[i] = dp[i - 1] + dp[i - 2]; // Build solution from bottom up
        }

        return dp[n]; // Final result
    }

    public static void main(String[] args) {
        int n = 10;
        System.out.println("Fibonacci of " + n + " is: " + fibonacci(n));
    }
}

এখানে, dp[] অ্যারে ব্যবহার করে bottom-up পদ্ধতিতে সমাধান তৈরি করা হয়েছে। ছোট ছোট সাব-প্রোঅব্লেম থেকে বড় সমস্যার সমাধান বের করা হচ্ছে এবং প্রতিটি সাব-প্রোঅব্লেমের ফলাফল সংরক্ষিত হচ্ছে।


Dynamic Programming এর সুবিধা এবং প্রয়োগ

সুবিধা:

  1. দ্রুত সমাধান: পুনরাবৃত্তি সমস্যা কমানো হয় এবং একই সাব-প্রোঅব্লেমের সমাধান বারবার গণনা না করতে হলে সমস্যা দ্রুত সমাধান হয়।
  2. প্রযুক্তিগত উন্নতি: এটি অনেক জটিল সমস্যার সমাধান দক্ষভাবে করতে সাহায্য করে, যেমন নেটওয়ার্ক ডিজাইন, অপটিমাইজেশন ইত্যাদি।

প্রয়োগ:

  1. ফিবোনাচ্চি সিকোয়েন্স
  2. Knapsack Problem (কনভিনিয়েন্ট প্যাকিং)
  3. Longest Common Subsequence (LCS)
  4. Matrix Chain Multiplication (মেট্রিক্স চেইন গুণন)
  5. Shortest Path Algorithms (ডিজিক্সট্রা, ফ্লয়েড-ওয়ারশাল)

সারাংশ

ডাইনামিক প্রোগ্রামিং একটি শক্তিশালী কৌশল যা পুনরাবৃত্তি সাব-প্রোঅব্লেমগুলির সমাধানকে সংরক্ষণ করে এবং অপটিমাইজেশন করতে সাহায্য করে। Overlapping Subproblems এবং Optimal Substructure এর মাধ্যমে সমস্যাকে ছোট ছোট অংশে ভাগ করে দ্রুত সমাধান করা সম্ভব। Top-Down (Memoization) এবং Bottom-Up (Tabulation) পদ্ধতিতে ডাইনামিক প্রোগ্রামিং প্রয়োগ করা যেতে পারে। এটি জটিল সমস্যাগুলোর সমাধানে বিশেষভাবে কার্যকরী, যেমন ফিবোনাচ্চি সিকোয়েন্স, কনভিনিয়েন্ট প্যাকিং, এবং শর্তাধীন পথ খোঁজা ইত্যাদি।


Content added By

Memoization এবং Tabulation হলো Dynamic Programming (ডাইনামিক প্রোগ্রামিং) এর দুটি প্রধান কৌশল যা সাধারণত recursive algorithms বা পুনরাবৃত্তি অ্যালগরিদমের দক্ষতা বাড়াতে ব্যবহৃত হয়। এই কৌশলগুলি একে অপরের বিপরীত হলেও তাদের লক্ষ্য একই: গণনা করা ফলাফলগুলো সংরক্ষণ করা যাতে পুনরায় একই ফলাফল গণনা না করতে হয়, যা পারফরম্যান্স উন্নত করে।

Memoization

Memoization হলো একটি উপায় যা top-down পদ্ধতিতে কাজ করে। এটি পুনরাবৃত্তি ফাংশনগুলিতে ফলাফলগুলো সংরক্ষণ করে যাতে পরবর্তীতে সেই ফলাফলগুলো পুনরায় গণনা করতে না হয়।

Tabulation

Tabulation হলো একটি bottom-up পদ্ধতি, যেখানে আপনি ছোট ছোট উপসমস্যাগুলির সমাধান করতে করতে বড় সমস্যার সমাধান করেন এবং একে একে সকল ফলাফল একটি টেবিল (array বা 2D array) এ সংরক্ষণ করেন।


১. Memoization (Top-Down)

Memoization হল একটি রিকার্সিভ কৌশল যা শুধুমাত্র প্রয়োজনীয় ফলাফল গুলি সংরক্ষণ করে। যখন একই সমস্যার জন্য পুনরায় গণনা করার প্রয়োজন হয় না, তখন পূর্বের ফলাফল ব্যবহার করা হয়।

উদাহরণ: Fibonacci Sequence - Memoization

ফিবোনাচ্চি সিকোয়েন্সের একটি সহজ উদাহরণ যেখানে আমরা Memoization ব্যবহার করে ফিবোনাচ্চি সিকোয়েন্স বের করার চেষ্টা করব।

import java.util.HashMap;

public class FibonacciMemoization {
    // HashMap ব্যবহার করে memoization সংরক্ষণ করা
    private HashMap<Integer, Integer> memo = new HashMap<>();

    // ফিবোনাচ্চি সিকোয়েন্সের জন্য মেমোইজড ফাংশন
    public int fibonacci(int n) {
        // যদি ফলাফল মেমোতে থাকে, তাহলে সোজা রিটার্ন করা
        if (memo.containsKey(n)) {
            return memo.get(n);
        }

        // বেস কেস
        if (n <= 1) {
            return n;
        }

        // Fibonacci সিকোয়েন্সের মান বের করা
        int result = fibonacci(n - 1) + fibonacci(n - 2);

        // ফলাফল মেমোতে সংরক্ষণ করা
        memo.put(n, result);

        return result;
    }

    public static void main(String[] args) {
        FibonacciMemoization fib = new FibonacciMemoization();
        int n = 10;
        System.out.println("Fibonacci of " + n + " is " + fib.fibonacci(n));
    }
}

ব্যাখ্যা:

  • Memoization দ্বারা fibonacci ফাংশনে পূর্বে গণনা করা মানগুলো সংরক্ষণ করা হয় memo হ্যাশম্যাপে।
  • প্রতিবার যখন একি মানের জন্য পুনরাবৃত্তি করা হয়, তখন Memoization সেই মানের জন্য পূর্বের ফলাফল সরাসরি রিটার্ন করে, ফলে গুণগতভাবে অপ্টিমাইজড করা হয়।

আউটপুট:

Fibonacci of 10 is 55

২. Tabulation (Bottom-Up)

Tabulation হল একটি bottom-up পদ্ধতি যেখানে আপনি ছোট ছোট সাবপ্রব্লেমগুলির জন্য ফলাফল একে একে গণনা করে একটি টেবিল বা অ্যারে ব্যবহার করে সেগুলিকে সংরক্ষণ করেন এবং পরে সেগুলিকে একত্রিত করে মূল সমস্যার সমাধান বের করেন।

উদাহরণ: Fibonacci Sequence - Tabulation

এখন, ফিবোনাচ্চি সিকোয়েন্সে Tabulation ব্যবহার করে সমাধান করা।

public class FibonacciTabulation {

    public int fibonacci(int n) {
        // Base cases for 0 and 1
        if (n <= 1) {
            return n;
        }

        // Tabulation: একটি অ্যারে তৈরি করা যা ফিবোনাচ্চি মান সংরক্ষণ করবে
        int[] dp = new int[n + 1];

        // বেস কেস
        dp[0] = 0;
        dp[1] = 1;

        // সিকোয়েন্সের মান গণনা করা
        for (int i = 2; i <= n; i++) {
            dp[i] = dp[i - 1] + dp[i - 2];  // Fibonacci relation
        }

        return dp[n];
    }

    public static void main(String[] args) {
        FibonacciTabulation fib = new FibonacciTabulation();
        int n = 10;
        System.out.println("Fibonacci of " + n + " is " + fib.fibonacci(n));
    }
}

ব্যাখ্যা:

  • এখানে Tabulation ব্যবহার করা হয়েছে যাতে ফলাফলগুলো একটি অ্যারে dp তে সংরক্ষিত হয়।
  • আমরা প্রথমে বেস কেস নির্ধারণ করি (যেমন dp[0] = 0 এবং dp[1] = 1) এবং তারপর বাকী মানগুলি বটম-আপ পদ্ধতিতে গণনা করি।

আউটপুট:

Fibonacci of 10 is 55

Memoization vs Tabulation

  1. Memoization (Top-Down):
    • Recursion ব্যবহার করে।
    • যতক্ষন না কোন সাবপ্রব্লেম প্রয়োজন হয় ততক্ষন কোন গণনা করা হয় না।
    • মেমোরি খরচ একটু বেশি হতে পারে (স্ট্যাক মেমরি) কারণ এটি রিকার্সিভ ফাংশন ব্যবহার করে।
  2. Tabulation (Bottom-Up):
    • Iterative পদ্ধতি ব্যবহার করে, কোনো রিকার্সন নেই।
    • সমস্ত সাবপ্রব্লেমের ফলাফল একটি টেবিল (অথবা অ্যারে) এ সংরক্ষিত হয়।
    • মেমোরি ব্যবহারে একটু কম, কারণ এটি স্ট্যাক ব্যবহার করে না।

সারাংশ

Memoization এবং Tabulation হলো Dynamic Programming এর দুটি মৌলিক কৌশল যা পুনরাবৃত্তি অ্যালগরিদমগুলির কার্যকারিতা উন্নত করতে সহায়তা করে।

  • Memoization: Top-Down পদ্ধতি, যেখানে আপনি রিকার্সিভ পদ্ধতিতে সাবপ্রব্লেমের ফলাফল সংরক্ষণ করেন।
  • Tabulation: Bottom-Up পদ্ধতি, যেখানে আপনি একে একে সব সাবপ্রব্লেম সমাধান করে একটি টেবিল তৈরি করেন।

এই কৌশলগুলির মাধ্যমে আমরা একই সাবপ্রব্লেম পুনরায় গণনা থেকে বাঁচাতে পারি এবং সময়ের দক্ষতা বৃদ্ধি করতে পারি।

Content added By

১. Longest Common Subsequence (LCS)

Longest Common Subsequence (LCS) হল দুটি স্ট্রিং বা সিকোয়েন্সের মধ্যে এমন একটি দীর্ঘতম উপসিকোয়েন্স (subsequence) খোঁজা যা উভয় সিকোয়েন্সে বিদ্যমান থাকে এবং যেটি তাদের মধ্যে অর্ডার অনুসরণ করে।

উদাহরণ:

  • স্ট্রিং 1: "ABCBDAB"
  • স্ট্রিং 2: "BDCABB"
  • LCS: "BCAB" (Longest Common Subsequence)

LCS Problem সাধারণত Dynamic Programming পদ্ধতিতে সমাধান করা হয়, যেখানে গ্রিড বা টেবিল তৈরি করে পূর্ববর্তী ফলাফল সংরক্ষণ করা হয়, যাতে একই সাব-সমস্যার পুনরাবৃত্তি না হয়।

উদাহরণ: LCS Algorithm in Java

public class LCS {
    public static int lcs(String str1, String str2) {
        int m = str1.length();
        int n = str2.length();
        
        // Create a table to store the lengths of longest common subsequence
        int[][] dp = new int[m + 1][n + 1];

        // Fill the table using dynamic programming approach
        for (int i = 1; i <= m; i++) {
            for (int j = 1; j <= n; j++) {
                if (str1.charAt(i - 1) == str2.charAt(j - 1)) {
                    dp[i][j] = dp[i - 1][j - 1] + 1; // If characters match
                } else {
                    dp[i][j] = Math.max(dp[i - 1][j], dp[i][j - 1]); // If characters don't match
                }
            }
        }

        return dp[m][n]; // The length of LCS is stored in dp[m][n]
    }

    public static void main(String[] args) {
        String str1 = "ABCBDAB";
        String str2 = "BDCABB";
        
        int result = lcs(str1, str2);
        System.out.println("Length of Longest Common Subsequence: " + result);
    }
}

ব্যাখ্যা:

  • Dynamic Programming Table: এখানে একটি 2D অ্যারে dp ব্যবহার করা হয়েছে, যেখানে dp[i][j] মানটি প্রথম i চরিত্রের জন্য এবং প্রথম j চরিত্রের জন্য LCS এর দৈর্ঘ্য সংরক্ষণ করে।
  • Recurrence Relation: যদি str1[i-1] == str2[j-1], তাহলে dp[i][j] = dp[i-1][j-1] + 1 (মানে তারা একটি লম্বা কমন সাবসিকোয়েন্স তৈরি করছে)। অন্যথায়, আমরা dp[i-1][j] এবং dp[i][j-1] এর মধ্যে বড় মান নেবো।

আউটপুট:

Length of Longest Common Subsequence: 4

Time Complexity: O(m * n), যেখানে m এবং n হল দুটি স্ট্রিংয়ের দৈর্ঘ্য।
Space Complexity: O(m * n), কারণ 2D টেবিল তৈরি করতে হয়।


২. Knapsack Problem

Knapsack Problem হল একটি Optimization Problem যেখানে আমাদের একটি নির্দিষ্ট ধারণক্ষমতা (capacity) সহ একটি বাক্স বা ব্যাগ (knapsack) দেওয়া থাকে এবং বিভিন্ন ওজন ও মানের বস্তু দেওয়া থাকে। আমাদের লক্ষ্য হল, সেই বস্তুগুলো নির্বাচন করা যেগুলির সম্মিলিত মান সর্বাধিক হয় এবং বস্তুগুলোর মোট ওজন দেওয়া ধারণক্ষমতার মধ্যে থাকে।

সমস্যার ধরন:

  1. 0/1 Knapsack Problem: প্রতি বস্তুকে একবারই বেছে নেওয়া যেতে পারে।
  2. Fractional Knapsack Problem: বস্তুগুলোর ভগ্নাংশও নেওয়া যেতে পারে।

এখানে আমরা 0/1 Knapsack Problem এর সমাধান করব, যা Dynamic Programming ব্যবহার করে সমাধান করা হয়।

উদাহরণ: 0/1 Knapsack Problem in Java

public class Knapsack {
    // Dynamic Programming solution for 0/1 Knapsack Problem
    public static int knapSack(int W, int[] wt, int[] val, int n) {
        int[][] dp = new int[n + 1][W + 1];

        // Build the table dp[][] in bottom up manner
        for (int i = 0; i <= n; i++) {
            for (int w = 0; w <= W; w++) {
                if (i == 0 || w == 0) {
                    dp[i][w] = 0; // If no items or no capacity, value is 0
                } else if (wt[i - 1] <= w) {
                    // If the item can be included in the knapsack
                    dp[i][w] = Math.max(val[i - 1] + dp[i - 1][w - wt[i - 1]], dp[i - 1][w]);
                } else {
                    // If the item can't be included in the knapsack
                    dp[i][w] = dp[i - 1][w];
                }
            }
        }

        return dp[n][W]; // Return the maximum value that can be achieved
    }

    public static void main(String[] args) {
        int[] val = {60, 100, 120}; // Values (profits)
        int[] wt = {10, 20, 30};   // Weights
        int W = 50;                 // Knapsack capacity
        int n = val.length;         // Number of items

        int result = knapSack(W, wt, val, n);
        System.out.println("Maximum value in Knapsack: " + result);
    }
}

ব্যাখ্যা:

  • Dynamic Programming Table: 2D অ্যারে dp[i][w] ব্যবহার করা হয়েছে, যেখানে dp[i][w] হল প্রথম i বস্তু দিয়ে knapsack এর ধারণক্ষমতা w এর জন্য সর্বাধিক মান।
  • Recurrence Relation:
    • যদি wt[i-1] <= w (যদি বর্তমান বস্তুটি knapsack এ ফিট হয়), তাহলে dp[i][w] = max(val[i-1] + dp[i-1][w-wt[i-1]], dp[i-1][w])
    • অন্যথায়, dp[i][w] = dp[i-1][w]

আউটপুট:

Maximum value in Knapsack: 220

Time Complexity: O(n * W), যেখানে n হল বস্তুগুলোর সংখ্যা এবং W হল knapsack এর ধারণক্ষমতা।
Space Complexity: O(n * W), কারণ একটি 2D ডিপি টেবিল ব্যবহার করা হয়।


সারাংশ

Longest Common Subsequence (LCS) এবং Knapsack Problem হল দুটি ক্লাসিক Dynamic Programming সমস্যার উদাহরণ, যেখানে প্রতিটি সমস্যা কিছু উপাদান নির্বাচন করে (যেমন, লংগেস্ট কমন সাবসিকোয়েন্স অথবা বস্তু নির্বাচন) এবং পূর্ববর্তী সিদ্ধান্তগুলির উপর ভিত্তি করে সর্বাধিক মান বা সমাধান খোঁজা হয়।

  • LCS দুইটি স্ট্রিংয়ের মধ্যে সবচেয়ে দীর্ঘ সাধারণ উপসিকোয়েন্স বের করে।
  • Knapsack Problem সীমিত ধারণক্ষমতায় সর্বাধিক মান সম্পন্ন বস্তু নির্বাচন করার সমস্যাকে সমাধান করে।

এই দুটি অ্যালগরিদমের মাধ্যমে, আপনি আরও জটিল সমস্যাগুলির জন্য Dynamic Programming কৌশল ব্যবহার করে কার্যকরী সমাধান পেতে পারেন।

Content added By

Dynamic Programming (DP) হল একটি শক্তিশালী অ্যালগরিদমিক টেকনিক যা সমস্যা সমাধানের জন্য ছোট সাব-প্রব্লেম গুলি পুনরায় ব্যবহার (overlapping subproblems) এবং সাব-অপ্টিমাল সলিউশনের পুনঃব্যবহার (optimal substructure) এর ধারণা ব্যবহার করে। এটি সাধারণত সমস্যা সমাধানে সময় ও স্পেসের জটিলতা কমাতে সাহায্য করে। Java তে Dynamic Programming (DP) সমস্যা সমাধান করতে কিছু সেরা অভ্যাস এবং কৌশল রয়েছে, যা অ্যালগরিদমগুলির কার্যকারিতা এবং রিডেবিলিটি উন্নত করতে সহায়ক।

এই টিউটোরিয়ালে, আমরা Dynamic Programming এর জন্য কিছু Best Practices এবং কিভাবে Java তে DP সমস্যাগুলি সমাধান করা যায় তা আলোচনা করব।


1. Sub-problems এবং Overlapping Sub-problems বুঝে কাজ করুন

Dynamic Programming তে সমস্যা সমাধানের জন্য প্রথমেই গুরুত্বপূর্ণ হল সমস্যার sub-problems চিহ্নিত করা। যদি একটি সমস্যা ছোট ছোট উপসমস্যায় ভাগ করা যায় এবং উপসমস্যাগুলির ফলাফল পুনরায় ব্যবহৃত হয়, তবে এটি একটি ক্লাসিক Dynamic Programming সমস্যা।

Best Practice:

  • উপসমস্যাগুলিকে যতটা সম্ভব ছোট রাখুন, এবং যেখানেই সম্ভব উপসমস্যার ফলাফলগুলিকে সঞ্চয় করুন (Memoization বা Tabulation ব্যবহার করে)।
  • কোনো উপসমস্যার ফলাফল একাধিক বার ব্যবহৃত হলে, তার ফলাফল ক্যালকুলেট করার পরিবর্তে সঞ্চয় করুন এবং সেগুলি পুনঃব্যবহার করুন।

2. Memoization এবং Tabulation এর মধ্যে সঠিক পছন্দ করুন

Memoization এবং Tabulation হল দুটি প্রধান কৌশল, যা DP সমস্যার সমাধানে ব্যবহৃত হয়:

  1. Memoization: এটি Top-down কৌশল, যেখানে ফাংশনটি রিকার্সিভভাবে ডাকা হয় এবং প্রতিটি সাব-প্রব্লেমের ফলাফল সঞ্চয় করা হয়।
  2. Tabulation: এটি Bottom-up কৌশল, যেখানে সমস্ত সম্ভাব্য সাব-প্রব্লেম সিস্টেম্যাটিকভাবে একটি টেবিলের মধ্যে সঞ্চিত হয়, এবং পরবর্তীতে বড় প্রব্লেম সমাধান করা হয়।

Best Practice:

  • Memoization ব্যবহার করা যেতে পারে যদি সাব-প্রব্লেমগুলির মধ্যে অল্প পুনঃব্যবহার থাকে।
  • যদি সমস্ত সাব-প্রব্লেম একে অপরের উপর নির্ভরশীল হয়, তবে Tabulation (Bottom-up) কৌশল ব্যবহার করা ভালো।

উদাহরণ: Fibonacci Numbers

Memoization (Top-down)
import java.util.HashMap;

public class FibonacciMemoization {
    private static HashMap<Integer, Integer> memo = new HashMap<>();

    public static int fibonacci(int n) {
        if (n <= 1) return n;

        if (memo.containsKey(n)) {
            return memo.get(n);
        }

        int result = fibonacci(n - 1) + fibonacci(n - 2);
        memo.put(n, result);
        return result;
    }

    public static void main(String[] args) {
        System.out.println(fibonacci(10));  // Output: 55
    }
}
Tabulation (Bottom-up)
public class FibonacciTabulation {
    public static int fibonacci(int n) {
        if (n <= 1) return n;

        int[] dp = new int[n + 1];
        dp[0] = 0;
        dp[1] = 1;

        for (int i = 2; i <= n; i++) {
            dp[i] = dp[i - 1] + dp[i - 2];
        }

        return dp[n];
    }

    public static void main(String[] args) {
        System.out.println(fibonacci(10));  // Output: 55
    }
}

ব্যাখ্যা:

  • Memoization: এখানে সাব-প্রব্লেমের ফলাফল HashMap এ সঞ্চিত থাকে, যা রিকার্সিভ ডাকা প্রতিবার ফলাফল প্রাপ্তি দ্রুত করে।
  • Tabulation: এখানে একটি অ্যারে তৈরি করা হয়, যেখানে সমস্ত সাব-প্রব্লেমের ফলাফল সঞ্চিত থাকে এবং পরবর্তীতে পরবর্তী ফলাফলগুলি আগের ফলাফলের উপর নির্ভর করে পূর্ণ হয়।

3. Space Optimization Techniques ব্যবহার করুন

Dynamic Programming এ অধিকাংশ সময় Tabulation ব্যবহার করলে বড় অ্যারে বা টেবিল তৈরি করতে হয়, যা অনেক সময় স্পেসের সমস্যার সৃষ্টি করতে পারে। আপনি শুধুমাত্র পূর্ববর্তী উপাদানগুলো মেমোরিতে রাখার মাধ্যমে স্পেস অপ্টিমাইজেশন করতে পারেন।

Best Practice:

  • বেশিরভাগ DP সমস্যায়, শুধুমাত্র অতীতের কিছু অবস্থান প্রয়োজন হয়। সেক্ষেত্রে space optimization কৌশল ব্যবহার করা যেতে পারে, যেখানে শুধুমাত্র প্রয়োজনীয় ডেটা রাখা হয় এবং পূর্ববর্তী সাব-প্রব্লেমের ফলাফলগুলো মুছে ফেলা হয়।

উদাহরণ: Fibonacci Numbers with Space Optimization

public class FibonacciSpaceOptimization {
    public static int fibonacci(int n) {
        if (n <= 1) return n;

        int prev1 = 0, prev2 = 1;
        for (int i = 2; i <= n; i++) {
            int current = prev1 + prev2;
            prev1 = prev2;
            prev2 = current;
        }
        return prev2;
    }

    public static void main(String[] args) {
        System.out.println(fibonacci(10));  // Output: 55
    }
}

ব্যাখ্যা:

  • এখানে শুধুমাত্র পূর্ববর্তী দুটি ফলাফল রাখা হয়েছে (prev1, prev2), যা স্পেসের ব্যবহার কমায় এবং O(1) স্পেসে Fibonacci সিরিজ গণনা করে।

4. Order of Solving Sub-problems ঠিক করুন

Dynamic Programming তে সঠিক সাব-প্রব্লেমের সমাধান করার জন্য উপযুক্ত ক্রম নির্ধারণ করা খুবই গুরুত্বপূর্ণ। কিছু DP সমস্যা top-down পদ্ধতিতে সমাধান করা উপযুক্ত (যেমন, Memoization) এবং কিছু সমস্যা bottom-up পদ্ধতিতে সমাধান করা উপযুক্ত (যেমন, Tabulation)।

Best Practice:

  • Top-down (Memoization) পদ্ধতিতে সাব-প্রব্লেমগুলি যতটা সম্ভব রিকার্সিভভাবে সমাধান করুন।
  • Bottom-up (Tabulation) পদ্ধতিতে পূর্বের ফলাফলগুলির উপর নির্ভরশীল সাব-প্রব্লেমগুলো একে একে সমাধান করুন।

5. Sub-problems এর ফলাফল গুলি Reuse করুন (Avoid Redundant Calculations)

Dynamic Programming এর মূল ধারণা হল সাব-প্রব্লেমের ফলাফল গুলি পুনঃব্যবহার করা। অতএব, সাব-প্রব্লেমের ফলাফলগুলি সঞ্চয় করা উচিত, যাতে তা আবার গণনা করার প্রয়োজন না হয়।

Best Practice:

  • Memoization অথবা Tabulation ব্যবহার করে সাব-প্রব্লেমের ফলাফল সঞ্চয় করুন।
  • যেকোনো ধরনের সমস্যায় যেখানে একাধিক সাব-প্রব্লেমের পুনঃব্যবহার হয়, সেখানে Memoization বা Tabulation একটি অপরিহার্য কৌশল।

6. Clear and Consistent Indexing ব্যবহার করুন

Dynamic Programming সমস্যা সমাধান করার সময়, আপনাকে সাব-প্রব্লেমের জন্য সঠিক ইনডেক্সিং নির্ধারণ করতে হবে। ভুল ইনডেক্সিং সমস্যার সমাধানে বিভ্রান্তি তৈরি করতে পারে।

Best Practice:

  • আপনার সমস্যা অনুযায়ী ডিপি টেবিল বা অ্যারে ইনডেক্সিং সঠিকভাবে ডিজাইন করুন। যেমন, যদি আপনি 2D টেবিল ব্যবহার করছেন, তবে আপনাকে 2টি ইনডেক্সের জন্য valid ranges নির্ধারণ করতে হবে।

7. Pruning Techniques ব্যবহার করুন

কিছু সমস্যা সমাধানে, pruning কৌশল ব্যবহার করা যেতে পারে, যেখানে আপনি সম্ভবত অপ্রয়োজনীয় সাব-প্রব্লেম সমাধান থেকে বিরত থাকতে পারেন।

Best Practice:

  • যেগুলি পুনরায় ব্যবহৃত হবে না বা কেবল সঠিক ফলাফলে পৌঁছানোর জন্য কেবলমাত্র একটি নির্দিষ্ট পথ প্রয়োজন, সেখানে pruning ব্যবহার করুন।

সারাংশ

Dynamic Programming (DP) হল একটি শক্তিশালী অ্যালগরিদমিক কৌশল যা উপসমস্যাগুলির ফলাফল পুনরায় ব্যবহার (overlapping subproblems) এবং সাব-অপ্টিমাল সমাধান পুনঃব্যবহার (optimal substructure) এর ধারণা ব্যবহার করে। Java তে DP সমস্যা সমাধানের জন্য কিছু সেরা অভ্যাস:

  1. সাব-প্রব্লেম গুলি সঠিকভাবে চিহ্নিত করুন।
  2. Memoization এবং Tabulation কৌশলগুলি ব্যবহার করুন।
  3. স্পেস অপ্টিমাইজেশন কৌশল ব্যবহার করুন।
  4. সাব-প্রব্লেম সমাধান করার সঠিক ক্রম নির্ধারণ করুন।
  5. ফলাফল পুনঃব্যবহার করুন এবং অপ্রয়োজনীয় গণনা এড়িয়ে চলুন।
  6. ইনডেক্সিং সঠিকভাবে করুন এবং প্রয়োজনে pruning কৌশল ব্যবহার করুন।

এই টিপসগুলি অনুসরণ করলে, আপনি Dynamic Programming এর সাহায্যে কার্যকরী ও দক্ষ অ্যালগরিদম তৈরি করতে সক্ষম হবেন।

Content added By
Promotion

Are you sure to start over?

Loading...