Computer Programming Greedy Algorithms (গ্রীডি এলগরিদম) গাইড ও নোট

529

গ্রীডি এলগরিদম (Greedy Algorithms)

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

গ্রীডি এলগরিদমটি সবচেয়ে ভাল কাজ করে এমন সমস্যা গুলোর জন্য যেখানে অপটিমাল সাবস্ট্রাকচার (Optimal Substructure) এবং গ্রেডিয়েন্টের উপকারিতা রয়েছে। এর মধ্যে, প্রতিটি সিদ্ধান্ত গ্রহণ করার সময়ে সামগ্রিক সমস্যার সবচেয়ে ভাল সমাধান পাওয়া যায়।

গ্রীডি এলগরিদমের মূল বৈশিষ্ট্যসমূহ:

  1. তৈরি সিদ্ধান্তে মনোযোগ: গ্রীডি এলগরিদম প্রতিটি ধাপে এমন একটি সিদ্ধান্ত নেয় যা তখনই সবচেয়ে ভালো বলে মনে হয়।
  2. অংশের সেরা নির্বাচন: প্রতিটি ছোট অংশের সেরা সমাধান নির্বাচন করা হয়, তবে সেটা পুরো সমস্যার জন্য সর্বোত্তম সমাধান নাও হতে পারে।
  3. পরবর্তী পদক্ষেপে সরাসরি প্রভাব: প্রতিটি সিদ্ধান্তের পরে পরবর্তী পদক্ষেপে কোনো পরিবর্তন না হয়, অর্থাৎ সোজাসুজি পরবর্তী অংশের জন্য সুবিধা তৈরি করতে চায়।

গ্রীডি এলগরিদমের ব্যবহারযোগ্যতা:

গ্রীডি এলগরিদমগুলি হ্যামিলটনিয়ান সমস্যা, ট্রাভেলিং সেলসম্যান প্রোবলেম (TSP), কুইলিং সমস্যার সমাধান, মিনিমাম স্প্যানিং ট্রি (MST), ক্যাশিং সমস্যা, ডাইনামিক প্রোগ্রামিং, ফ্লাইট রুটিং ইত্যাদিতে ব্যবহৃত হয়।


গ্রীডি এলগরিদমের উদাহরণ:

১. মিনিমাম স্প্যানিং ট্রি (Minimum Spanning Tree - MST)

মিনিমাম স্প্যানিং ট্রি হল এমন একটি ট্রি যেখানে গ্রাফের সমস্ত ভেরটেক্স সংযুক্ত থাকে এবং মোট ওজন (weight) সর্বনিম্ন থাকে। দুটি জনপ্রিয় গ্রীডি এলগরিদম যা MST তৈরি করতে ব্যবহৃত হয় তা হল প্রাইম এলগরিদম (Prim's Algorithm) এবং **ক্রাসকাল এলগরিদম (Kruskal's Algorithm)**।

প্রাইম এলগরিদম (Prim's Algorithm):

এটি একটি গ্রীডি এলগরিদম যা একটি গ্রাফের একটি শীর্ষ থেকে শুরু করে, যতক্ষণ না সমস্ত শীর্ষ সংযুক্ত হয়, এমনভাবে কাজ করে।

প্রাইম এলগরিদম সি কোড:

#include <stdio.h>
#include <limits.h>

#define V 5  // Number of vertices

// Function to find the vertex with the minimum key value
int minKey(int key[], int mstSet[]) {
    int min = INT_MAX, minIndex;

    for (int v = 0; v < V; v++) {
        if (mstSet[v] == 0 && key[v] < min) {
            min = key[v];
            minIndex = v;
        }
    }
    return minIndex;
}

// Function to implement Prim's algorithm
void primMST(int graph[V][V]) {
    int parent[V];  // Array to store constructed MST
    int key[V];     // Key values to pick the minimum weight edge
    int mstSet[V];  // To represent vertices that are included in MST

    // Initialize all keys as INFINITE
    for (int i = 0; i < V; i++) {
        key[i] = INT_MAX;
        mstSet[i] = 0;
    }

    // Always include the first vertex in MST
    key[0] = 0;
    parent[0] = -1;

    // Construct the MST
    for (int count = 0; count < V - 1; count++) {
        int u = minKey(key, mstSet);  // Get the vertex with the minimum key value

        mstSet[u] = 1;  // Add vertex u to MST

        // Update the key values and parent indices of adjacent vertices
        for (int v = 0; v < V; v++) {
            if (graph[u][v] && mstSet[v] == 0 && graph[u][v] < key[v]) {
                parent[v] = u;
                key[v] = graph[u][v];
            }
        }
    }

    // Print the constructed MST
    printf("Edge \tWeight\n");
    for (int i = 1; i < V; i++) {
        printf("%d - %d \t%d\n", parent[i], i, graph[i][parent[i]]);
    }
}

int main() {
    // Example graph
    int graph[V][V] = {
        {0, 2, 0, 6, 0},
        {2, 0, 3, 8, 5},
        {0, 3, 0, 0, 7},
        {6, 8, 0, 0, 9},
        {0, 5, 7, 9, 0}
    };

    // Call Prim's algorithm
    primMST(graph);

    return 0;
}

ব্যাখ্যা:

  • এখানে, প্রাইম এলগরিদম ব্যবহার করে গ্রাফের মিনিমাম স্প্যানিং ট্রি তৈরি করা হয়েছে।
  • key[] অ্যারে গ্রাফের প্রতিটি ভেরটেক্সের জন্য সবচেয়ে কম ওজনের এজ ধরে রাখে।
  • mstSet[] অ্যারে ট্র্যাক করে কোন ভেরটেক্স MST তে অন্তর্ভুক্ত হয়েছে।

২. হাফ-শপ প্রোবলেম (Fractional Knapsack Problem)

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

হাফ-শপ প্রোবলেম সি কোড:

#include <stdio.h>

// Structure to represent an item
struct Item {
    int weight;
    int value;
    double ratio;
};

// Function to compare two items according to their value/weight ratio
int compare(const void* a, const void* b) {
    double r1 = ((struct Item*)a)->ratio;
    double r2 = ((struct Item*)b)->ratio;
    return (r1 < r2) - (r1 > r2);
}

// Function to solve the Fractional Knapsack problem
double knapsack(int capacity, struct Item arr[], int n) {
    // Sort items by value/weight ratio
    qsort(arr, n, sizeof(arr[0]), compare);

    int totalWeight = 0;
    double totalValue = 0.0;

    // Loop through the items
    for (int i = 0; i < n; i++) {
        // If adding the whole item doesn't exceed capacity, add it
        if (totalWeight + arr[i].weight <= capacity) {
            totalWeight += arr[i].weight;
            totalValue += arr[i].value;
        } else {
            // If the item can't be added entirely, add a fraction of it
            int remainingWeight = capacity - totalWeight;
            totalValue += arr[i].value * ((double)remainingWeight / arr[i].weight);
            break;
        }
    }

    return totalValue;
}

int main() {
    struct Item arr[] = {{10, 60}, {20, 100}, {30, 120}};
    int n = sizeof(arr) / sizeof(arr[0]);
    int capacity = 50;

    printf("Maximum value we can obtain = %.2f\n", knapsack(capacity, arr, n));

    return 0;
}

ব্যাখ্যা:

  • এখানে হাফ-শপ প্রোবলেম সমাধান করতে গ্রীডি এলগরিদম ব্যবহার করা হয়েছে, যেখানে আইটেমের ভ্যালু/ওজন রেশিও অনুযায়ী আইটেমগুলি সাজানো হয়েছে।
  • যদি কোনো আইটেম পুরোপুরি ব্যাগে না ফিট হয়, তবে সেই আইটেমের একটি অংশ (ভগ্নাংশ) যোগ করা হয়।

গ্রীডি এলগরিদমের সুবিধা এবং অসুবিধা

সুবিধা:

  • গ্রীডি এলগরিদম খুব দ্রুত কাজ করে এবং অনেক ক্ষেত্রে গাণিতিকভাবে সহজ হয়।
  • এটি সহজভাবে কনসেপ্ট করা যায় এবং কার্যকরী সমাধান প্রদান করে।

অসুবিধা:

  • গ্রীডি এলগরিদম সব সময় সেরা সমাধান দেয় না, তবে এটি প্রায়ই প্রায় optima (near optimal) সমাধান প্রদান করে।
  • কিছু সমস্যায় গ্রীডি এলগরিদম ভুল সম

াধান দিতে পারে, যদি সমস্যা অপটিমাল সাবস্ট্রাকচার না থাকে।


সারসংক্ষেপ

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

Content added By

Greedy Algorithm এর ধারণা এবং এর প্রয়োজনীয়তা

371

Greedy Algorithm এর ধারণা এবং এর প্রয়োজনীয়তা

Greedy Algorithm (লালসামুখী অ্যালগরিদম) একটি সমস্যা সমাধানের কৌশল যেখানে প্রতিটি ধাপে সবচেয়ে ভালো বা সর্বাধিক উপকারী সিদ্ধান্ত নেওয়া হয়, তবে ভবিষ্যতের পরিস্থিতি বা পরবর্তী ধাপের জন্য কোনো রকম পুনর্বিবেচনা বা পরিকল্পনা করা হয় না। অর্থাৎ, Greedy Algorithm প্রতিটি ধাপে এমন একটি সিদ্ধান্ত নেয় যা তাকে এক্ষণিকভাবে সর্বোত্তম উপকার বা লাভ দেয়, কিন্তু এটি সর্বসম্মত বা সর্বশেষ সেরা সমাধান না-ও হতে পারে।


Greedy Algorithm এর মৌলিক ধারণা:

  1. সর্বোত্তম স্থানীয় সমাধান নির্বাচন:
    • Greedy Algorithm প্রতিটি ধাপে সর্বোত্তম স্থানীয় সমাধান বা local optimal solution বেছে নেয়, যা তার কাছে তৎক্ষণাৎ সর্বোত্তম মনে হয়। কিন্তু ভবিষ্যতের অবস্থার প্রেক্ষিতে এটি সর্বোত্তম সমাধান নাও হতে পারে।
  2. ফাইনাল গন্তব্যে পৌঁছানো:
    • প্রত্যেক পদক্ষেপে নেয়া সর্বোত্তম সিদ্ধান্তের ফলে সমগ্র প্রক্রিয়া শেষে globally optimal solution (সর্বোত্তম গ্লোবাল সমাধান) পাওয়া যাবে, এমন একটি আশা করা হয়।
  3. প্রত্যেক সিদ্ধান্তের পূর্ণতা:
    • একটি সিদ্ধান্ত নেয়ার পর, পরবর্তী সিদ্ধান্তের জন্য পূর্বের সিদ্ধান্তের ওপর কোনো পরিবর্তন বা পুনঃমূল্যায়ন করা হয় না, যা এটিকে দ্রুত কার্যকরী করে তোলে।
  4. সমস্যার বৈশিষ্ট্য:
    • Greedy Algorithm সফল হতে পারে যখন সমস্যাটি গ্রীডি প্রকৃতি (greedy property) এবং অপটিমাল সাবস্ট্রাকচার (optimal substructure) অনুসরণ করে।
      • গ্রীডি প্রকৃতি: স্থানীয়ভাবে সর্বোত্তম সিদ্ধান্ত গ্রহণ করলে সমগ্র সমস্যার জন্য সর্বোত্তম সমাধান পাওয়া যাবে।
      • অপটিমাল সাবস্ট্রাকচার: সমস্যার ছোট অংশগুলোর সমাধান একত্রিত করে মূল সমস্যার সমাধান করা যায়।

Greedy Algorithm এর প্রয়োজনীয়তা:

  1. দ্রুত সমাধান:
    • Greedy Algorithm সাধারণত খুব দ্রুত কাজ করে, কারণ এটি কোনো পুনরাবৃত্তি বা পরবর্তী সিদ্ধান্তে ফিরে যাওয়া ছাড়াই তৎক্ষণাত সর্বোত্তম সিদ্ধান্ত নেয়। এটি ছোট বা মাঝারি আকারের সমস্যা দ্রুত সমাধান করতে সহায়ক।
    • উদাহরণ: Fractional Knapsack Problem যেখানে আপনার প্রতি ধাপে সর্বাধিক লাভজনক বস্তু নির্বাচিত হয়।
  2. সহজ বাস্তবায়ন:
    • Greedy Algorithm এর কৌশলটি তুলনামূলকভাবে সহজ, এবং কোডিং বা বাস্তবায়নও অনেক সহজ হয়।
    • উদাহরণ: Huffman Coding, যেখানে প্রতিটি ধাপে সর্বনিম্ন ফ্রিকোয়েন্সি উপাদানটি নির্বাচন করা হয়।
  3. রিসোর্স অপ্টিমাইজেশন:
    • যেখানে একটি সমাধানে সীমিত রিসোর্সের জন্য কাজ করা হয়, সেখানে Greedy Algorithm একটি দ্রুত রিসোর্স অপ্টিমাইজেশন সমাধান প্রদান করে।
    • উদাহরণ: Activity Selection Problem, যেখানে সময়ের মধ্যে সর্বাধিক কাজ নির্বাচন করতে হয়।
  4. ব্যবহারিক সমস্যাগুলির সমাধান:
    • বাস্তব জীবনে অনেক সমস্যা এমন হতে পারে যেখানে, তৎক্ষণাৎ সেরা সমাধান নিয়ে এগিয়ে যাওয়া সম্ভব হয়। যেমন, ট্র্যাফিক নিয়ন্ত্রণ, নগদ অর্থ সংগ্রহ, পর্যটক সমস্যা ইত্যাদি।

Greedy Algorithm এর ব্যবহার:

  1. Knapsack Problem (Fractional Knapsack):
    • একটি বস্তু বা আইটেমের সঠিক অংশ নির্বাচন করা যা সবচেয়ে লাভজনক এবং সীমিত ক্যাপাসিটি বা রিসোর্স নিয়ে সম্পন্ন করা যায়। Fractional Knapsack এ Greedy Algorithm ব্যবহৃত হয়, যেখানে সবচেয়ে বেশি লাভ দেওয়া আইটেমটি প্রথমে নির্বাচিত হয়।
  2. Activity Selection Problem:
    • একটি নির্দিষ্ট সময়ে সবচেয়ে বেশি কার্যকলাপ নির্বাচন করতে হবে। যেখানে সেরা ফলাফল পেতে প্রতিটি পদক্ষেপে সর্বোত্তম সিদ্ধান্ত নেয়া হয়।
  3. Huffman Coding:
    • এটি একটি জনপ্রিয় আলগোরিদম যা ডেটা কমপ্রেশন এর জন্য ব্যবহৃত হয়। এখানে প্রতিটি ধাপে সর্বনিম্ন ফ্রিকোয়েন্সি যুক্ত চরিত্রগুলোকে একত্রিত করা হয়।
  4. Dijkstra's Algorithm:
    • গ্রাফের মধ্যে সেরা পাথ খুঁজে বের করার জন্য এই অ্যালগরিদম ব্যবহৃত হয়। এটি Greedy পদ্ধতির উপর ভিত্তি করে কাজ করে।
  5. Minimum Spanning Tree (MST):
    • গ্রাফের মধ্যে সব নোডকে সংযুক্ত করার জন্য Prim's Algorithm এবং Kruskal's Algorithm ব্যবহার করা হয়। এটি গ্রীডি অ্যালগরিদমের উদাহরণ যেখানে স্থানীয়ভাবে সর্বোত্তম সীমানা চয়ন করা হয়।

Greedy Algorithm এর উদাহরণ:

Activity Selection Problem:

ধরা যাক, আপনার কাছে একাধিক কাজ রয়েছে এবং প্রতিটি কাজের একটি শুরু এবং শেষ সময় আছে। আমাদের লক্ষ্য হলো, সর্বাধিক কাজ একে অপরের মধ্যে সময়ের সংঘর্ষ ছাড়া করতে।

Greedy Approach: কাজগুলি তাদের শেষ সময়ের অনুযায়ী সাজানো এবং সর্বনিম্ন সময়ের কাজ প্রথমে নির্বাচন করা হয়।

#include <stdio.h>

struct Activity {
    int start;
    int finish;
};

void activitySelection(struct Activity arr[], int n) {
    int i, j;

    printf("Selected activities: \n");

    // প্রথম কাজ নির্বাচন করা
    i = 0;
    printf("(%d, %d) ", arr[i].start, arr[i].finish);

    // অন্যান্য কাজ নির্বাচন করা
    for (j = 1; j < n; j++) {
        if (arr[j].start >= arr[i].finish) {
            printf("(%d, %d) ", arr[j].start, arr[j].finish);
            i = j;
        }
    }
}

int main() {
    struct Activity arr[] = {{1, 3}, {2, 5}, {4, 6}, {6, 8}, {5, 7}};
    int n = sizeof(arr) / sizeof(arr[0]);

    // কাজগুলোকে শেষ সময় অনুসারে সাজানো
    // Sort by finish time
    qsort(arr, n, sizeof(struct Activity), cmp);

    activitySelection(arr, n);
    return 0;
}

Greedy Algorithm এর সুবিধা এবং অসুবিধা:

সুবিধা:

  1. সহজ এবং দ্রুত: Greedy Algorithm সাধারণত দ্রুত এবং সহজে বাস্তবায়িত হতে পারে।
  2. অপারেশনের কম সময়: অধিকাংশ ক্ষেত্রেই Greedy Algorithm অন্যান্য অপটিমাইজেশন পদ্ধতির তুলনায় কম সময় নিয়ে কাজ করে।
  3. ব্যবহারিক সমস্যা সমাধান: বাস্তব জীবনের অনেক সমস্যায় Greedy Algorithm কার্যকরী হতে পারে, যেমন Activity Selection বা Huffman Coding

অসুবিধা:

  1. সবসময় সর্বোত্তম সমাধান নয়: Greedy Algorithm প্রতিটি ধাপে সর্বোত্তম সিদ্ধান্ত নেয়, কিন্তু তা সবসময় গ্লোবালভাবে সেরা সমাধান দিতে পারে না।
  2. পুনর্বিবেচনার অভাব: Greedy Algorithm বর্তমান সিদ্ধান্তের দিকে মনোনিবেশ করে এবং ভবিষ্যতের সম্ভাবনা পর্যালোচনা করা হয় না।
  3. মুলতঃ সহজ সমস্যা সমাধান: কিছু জটিল সমস্যায় Greedy Algorithm ভুল সিদ্ধান্ত নিতে পারে, যেখানে পুনর্বিবেচনার প্রয়োজন হয়।

সারসংক্ষেপ:

Greedy Algorithm এমন একটি কৌশল যা সমস্যার সমাধান করতে প্রতিটি ধাপে সর্বোত্তম স্থানীয় সিদ্ধান্ত নেয়। এটি Activity Selection Problem, Knapsack Problem, Huffman Coding এবং Minimum Spanning Tree ইত্যাদি বিভিন্ন সমস্যা সমাধানে ব্যবহৃত হয়। যদিও Greedy Algorithm অনেক সমস্যার জন্য দ্রুত এবং কার্যকরী হতে পারে, তবে এটি সব সময় globally optimal solution (গ্লোবালভাবে সর্বোত্তম সমাধান) প্রদান করতে সক্ষম হয় না।

Content added By

বাস্তব জীবনের সমস্যা সমাধানে Greedy Technique

312

বাস্তব জীবনের সমস্যা সমাধানে Greedy Technique

Greedy Algorithm (লোভী কৌশল) এমন একটি পদ্ধতি যা সমস্যার প্রতিটি ধাপে লোভী পছন্দ করে এবং সর্বোত্তম সমাধান অর্জন করতে চেষ্টা করে। এটি সর্বদা একটি স্থানীয়ভাবে সর্বোত্তম (locally optimal) সিদ্ধান্ত নেয়, যা শেষ পর্যন্ত একটি গ্লোবালি অপটিমাল (globally optimal) সমাধান হতে পারে বা নাও হতে পারে, তবে সাধারণত Greedy কৌশল সহজ এবং দ্রুত সমাধান দেয়। এই কৌশলটি অনেক বাস্তব জীবনের সমস্যায় কার্যকরী হয়।


Greedy Algorithm এর বৈশিষ্ট্য:

  1. স্থানীয়ভাবে সর্বোত্তম সিদ্ধান্ত: প্রতিটি পদক্ষেপে সেরা অপশনটি নির্বাচন করা হয়, এবং এটি সমস্যার সমাধান করতে সহায়তা করে।
  2. নতুন সিদ্ধান্তের উপর নির্ভরশীলতা: একে অপরের উপর ভিত্তি করে সিদ্ধান্তগুলো নেওয়া হয়।
  3. গ্লোবালি সর্বোত্তম সমাধান নয়: Greedy কৌশলটি সেরা সমাধান দেয় না সব সময়, তবে এটি খুব দ্রুত কাজ করতে সক্ষম।

Greedy Algorithm এর বাস্তব জীবনের সমস্যার উদাহরণ

  1. ক্যাশ চেঞ্জ সমস্যা (Coin Change Problem)

ক্যাশ চেঞ্জ বা কয়েন পরিবর্তন একটি ক্লাসিক উদাহরণ যেখানে একজন দোকানদারকে কোনো পরিমাণ টাকার পরিবর্তে কয়েন দেওয়া হয়। Greedy কৌশল ব্যবহার করলে আমরা প্রতিটি পদক্ষেপে সর্বোচ্চ মানের কয়েন বেছে নিয়ে পরিমাণ পূর্ণ করতে পারি।

সমস্যা:

ধরা যাক, আপনাকে 63 টাকা ফেরত দিতে হবে এবং কয়েনের মূল্য হচ্ছে 25, 10, 5, এবং 1। Greedy কৌশল অনুযায়ী, আপনি প্রথমে সর্বোচ্চ কয়েন (25) ব্যবহার করবেন, তারপর 10, 5, এবং অবশেষে 1। এটি দ্রুত এবং সঠিক সমাধান প্রদান করে।

Greedy কৌশলের পদক্ষেপ:

  • প্রথমে 25 টাকা দিয়ে শুরু করুন: 63 - 25 = 38
  • তারপর 25 টাকা আরও একটি কয়েন দিন: 38 - 25 = 13
  • তারপর 10 টাকা দিন: 13 - 10 = 3
  • তারপর 5 টাকা দিন (যদিও এটি বৃহত্তম নয়, কিন্তু পরবর্তী পর্যায়ে 1 টাকা প্রয়োজন): 3 - 3 = 0

এইভাবে সমাধানটি দ্রুত এবং কার্যকরী হয়। কিন্তু যদি কয়েনের মান ভিন্ন হতো, যেমন 1, 3, 4 এর মতো, তখন Greedy কৌশলটি সর্বোত্তম সমাধান নাও দিতে পারে।


  1. হাফ-টিমিং এবং কাজের শিডিউলিং (Job Scheduling Problem)

কাজের শিডিউলিং সমস্যা যেখানে কাজগুলির শেষ সময় এবং মুনাফা দেওয়া থাকে, এবং লক্ষ্য থাকে যে আপনি সর্বাধিক মুনাফা অর্জন করতে পারেন।

সমস্যা:

আপনি যদি বিভিন্ন কাজের একটি তালিকা পান, যেখানে প্রতিটি কাজের একটি নির্দিষ্ট সময়সীমা এবং মুনাফা দেওয়া থাকে, তাহলে Greedy কৌশল দ্বারা আপনাকে এমন কাজগুলো বেছে নিতে হবে যা সর্বোচ্চ মুনাফা দেবে এবং কোনো কাজের সাথে সময়ের সংঘর্ষ হবে না।

Greedy কৌশল:

  • প্রথমে সেই কাজগুলো বেছে নিন যেগুলোর মুনাফা সবচেয়ে বেশি, এবং সময়কালগুলোতে একে অপরকে ছাপিয়ে না যাওয়ার নিশ্চয়তা দিন।
  • এর মাধ্যমে আপনি সর্বোচ্চ মুনাফা পেতে পারবেন।

  1. হফম্যান কোডিং (Huffman Coding)

হফম্যান কোডিং একটি ডিজিটাল এনকোডিং স্কিমা যা তথ্য সংরক্ষণ করতে ব্যবহৃত হয়। এটি একটি প্রকারের Greedy কৌশল যেখানে দৈর্ঘ্যভিত্তিক কোড দেওয়া হয়, এবং সবার জন্য সেরা টিউনিং করতে ছোট ডেটার জন্য কম বিট এবং বড় ডেটার জন্য বেশি বিট ব্যবহৃত হয়।

Greedy কৌশল:

  • প্রতিটি চরিত্রের ফ্রিকোয়েন্সি হিসাব করুন।
  • সর্বনিম্ন ফ্রিকোয়েন্সির দুটি চরিত্রের জন্য একটি নতুন নোড তৈরি করুন এবং তাদেরকে একটি নতুন আর্বি ট্রি (Huffman Tree) হিসেবে সমন্বিত করুন।
  • এই প্রক্রিয়া চলতে থাকবে যতক্ষণ না আপনি একটি পূর্ণাঙ্গ হাফম্যান ট্রি পেয়ে যাবেন।

এই পদ্ধতিতে সেরা এনকোডিং পাওয়া যায় এবং এটি ডেটা কম্প্রেশনেও ব্যবহৃত হয়।


  1. প্রধান রাস্তা নির্মাণ (Minimum Spanning Tree - MST)

গ্রাফের মধ্যে সবগুলো নোড সংযুক্ত করতে এমন একটি গাছ (tree) খুঁজে বের করার সমস্যা যা মোট ওয়েট বা ব্যয় কমায়।

সমস্যা:

যেমন ধরুন, আপনি বিভিন্ন শহরকে রাস্তা দিয়ে সংযুক্ত করতে চান, কিন্তু মোট খরচ কম রাখতে চান।

Greedy কৌশল:

  • Kruskal's Algorithm এবং Prim's Algorithm দুটোই MST তৈরি করার জন্য Greedy কৌশল ব্যবহার করে।
    • Prim’s Algorithm: এটি একে একে গ্রাফের নোডগুলো যোগ করে এবং প্রতিটি পদক্ষেপে সর্বনিম্ন ব্যয়ের রাস্তাটি বেছে নেয়।
    • Kruskal’s Algorithm: এটি গ্রাফের সকল এজগুলোকে একে একে বেছে নেয় এবং সর্বনিম্ন ব্যয়ের এজগুলো সংযুক্ত করে।

এইভাবে Greedy কৌশল MST তৈরি করতে সক্ষম হয়।


  1. ডিক্সট্রা অ্যালগরিদম (Dijkstra’s Algorithm)

এটি একটি গ্রাফের মধ্যে সর্বনিম্ন পথ খুঁজে বের করার জন্য ব্যবহৃত হয়। বিশেষত, Greedy কৌশল ব্যবহার করে ডিক্সট্রা অ্যালগরিদম প্রতিটি নোডের জন্য সর্বনিম্ন ব্যয় (distance) নির্বাচন করে।

সমস্যা:

আপনি যদি একটি গ্রাফে একটি নোড থেকে অন্য নোডে যাওয়ার সর্বনিম্ন পথ খুঁজতে চান, তবে ডিক্সট্রা অ্যালগরিদম ব্যবহার করা হয়।

Greedy কৌশল:

  • প্রথমে সর্বনিম্ন দূরত্ব সহ একটি নোড নির্বাচন করা হয়, তারপর প্রতিটি ভিজিটেড নোডের জন্য সেই নোডে যাওয়ার নতুন রাস্তা খুঁজে বের করা হয়।
  • এই প্রক্রিয়া পুনরাবৃত্তি করা হয় যতক্ষণ না সব নোডের জন্য সর্বনিম্ন পথ পাওয়া না যায়।

সারসংক্ষেপ

  • Greedy Technique একটি সমস্যার প্রতিটি ধাপে সর্বোত্তম সিদ্ধান্ত নেয়, তবে এটি সর্বোত্তম গ্লোবাল সমাধান দেয় না সবসময়।
  • বাস্তব জীবনের বিভিন্ন সমস্যায় Greedy কৌশল কার্যকরী হয়, যেমন ক্যাশ চেঞ্জ সমস্যা, টাস্ক শিডিউলিং, হফম্যান কোডিং, গ্রাফের মিনিমাম স্প্যানিং ট্রি ইত্যাদি।
  • এটি সাধারণত দ্রুত এবং সহজ সমাধান প্রদান করে, তবে কখনও কখনও এটি গ্লোবাল অপটিমাল সলিউশন নিশ্চিত করে না।
Content added By

Activity Selection Problem, Fractional Knapsack Problem

283

Activity Selection Problem এবং Fractional Knapsack Problem দুটি গুরুত্বপূর্ণ গ্রীড-ভিত্তিক অপটিমাইজেশন সমস্যা। এই সমস্যা গুলি প্রায়ই গ্রীড বা কন্টিনিউয়াস ডেটা স্ট্রাকচারের মধ্যে সমাধান করা হয় এবং তাদের মধ্যে বেশ কিছু অ্যালগরিদম ব্যবহার করা হয়।


১. Activity Selection Problem

Activity Selection Problem হল একটি ক্লাসিকাল অপটিমাইজেশন সমস্যা যেখানে আপনাকে কিছু কর্মকাণ্ড (activities) দেওয়া হয় এবং আপনার কাজ হল, আপনি কতগুলো কর্মকাণ্ড একসাথে সম্পন্ন করতে পারেন এমনভাবে বেছে নেওয়া, যাতে তাদের মধ্যে সময়ের মধ্যে কোনও সম্পর্ক না থাকে (অর্থাৎ, একটির সময় অন্যটির সাথে ওভারল্যাপ না করে)।

Problem Description:

  • আপনাকে বিভিন্ন কর্মকাণ্ডের একটি তালিকা দেওয়া হয় যেখানে প্রতিটি কর্মকাণ্ডের একটি স্টার্ট টাইম এবং একটি এন্ড টাইম আছে।
  • আপনার কাজ হল এমন কর্মকাণ্ড নির্বাচন করা, যাতে তাদের মধ্যে কোনও সময়ের চূড়ান্ত অঙ্ক না থাকে এবং যত বেশি সম্ভব কর্মকাণ্ড নির্বাচিত হয়।

Solution Approach:

এই সমস্যা সমাধানের জন্য Greedy Algorithm ব্যবহার করা হয়। অ্যালগরিদমটি নিম্নলিখিত ধাপে কাজ করে:

  1. Activities Sort: প্রথমে সমস্ত কর্মকাণ্ডের এন্ড টাইম অনুযায়ী সাজানো হয়।
  2. প্রথম কর্মকাণ্ডটি নির্বাচন করুন এবং তারপর একে একে পরবর্তী কর্মকাণ্ডগুলো চেক করুন।
  3. যদি কোনো কর্মকাণ্ডের স্টার্ট টাইম পূর্ববর্তী নির্বাচিত কর্মকাণ্ডের এন্ড টাইম এর পরে থাকে, তবে সেটি নির্বাচন করুন।

Time Complexity:

  • Sorting: O(n log n)
  • Selection: O(n)
  • Total Time Complexity: O(n log n)

সি প্রোগ্রামে Activity Selection Problem এর ইমপ্লিমেন্টেশন:

#include <stdio.h>
#include <stdlib.h>

typedef struct {
    int start;
    int end;
} Activity;

int compare(const void *a, const void *b) {
    return ((Activity *)a)->end - ((Activity *)b)->end;
}

void selectActivities(Activity activities[], int n) {
    // Sort activities by end time
    qsort(activities, n, sizeof(Activity), compare);

    printf("Selected activities are:\n");
    
    // The first activity is always selected
    int lastSelected = 0;
    printf("Activity %d: [%d, %d]\n", 1, activities[lastSelected].start, activities[lastSelected].end);
    
    // Check remaining activities
    for (int i = 1; i < n; i++) {
        if (activities[i].start >= activities[lastSelected].end) {
            lastSelected = i;
            printf("Activity %d: [%d, %d]\n", i + 1, activities[lastSelected].start, activities[lastSelected].end);
        }
    }
}

int main() {
    Activity activities[] = {{1, 3}, {2, 5}, {4, 7}, {6, 8}, {5, 9}};
    int n = sizeof(activities) / sizeof(activities[0]);
    
    selectActivities(activities, n);

    return 0;
}

২. Fractional Knapsack Problem

Fractional Knapsack Problem একটি অপটিমাইজেশন সমস্যা যেখানে একটি স্লটের মধ্যে ডেটা সংরক্ষণ করতে আপনাকে কিছু আইটেম দেওয়া হয় এবং তাদের মধ্যে সর্বোচ্চ মূল্য (value) অর্জন করার জন্য, আপনি কিছু আইটেম সম্পূর্ণ বা আংশিকভাবে নিতে পারেন। এর মানে, আপনি আইটেমগুলোকে টুকরো টুকরো করে নিতে পারবেন (Fractional), যা 0/1 Knapsack Problem থেকে পৃথক।

Problem Description:

  • আপনি একটি ন্যূনতম ক্ষমতা সম্পন্ন ব্যাগের মধ্যে বিভিন্ন আইটেম নির্বাচন করবেন।
  • প্রতিটি আইটেমের একটি ওজন (weight) এবং একটি মূল্য (value) থাকবে।
  • আপনি যে কোন আইটেমের একাধিক অংশ নিতে পারবেন, অর্থাৎ Fractional Knapsack।
  • আপনার লক্ষ্য হলো, ব্যাগের মধ্যে সর্বোচ্চ মূল্য (value) সন্নিবেশ করা।

Solution Approach:

এই সমস্যাটি সমাধান করার জন্য Greedy Algorithm ব্যবহার করা হয়, যেখানে প্রথমে আইটেমগুলির মূল্য প্রতি ইউনিট ওজনের জন্য (value/weight) অনুপাত হিসাব করা হয়। এরপর আইটেমগুলোকে এই অনুপাত অনুযায়ী সাজানো হয়, এবং তাদের একে একে ব্যাগে সন্নিবেশ করা হয় যতক্ষণ না ব্যাগ পূর্ণ হয়।

Time Complexity:

  • Sorting: O(n log n)
  • Selection: O(n)
  • Total Time Complexity: O(n log n)

সি প্রোগ্রামে Fractional Knapsack Problem এর ইমপ্লিমেন্টেশন:

#include <stdio.h>
#include <stdlib.h>

typedef struct {
    int weight;
    int value;
    double ratio;
} Item;

int compare(const void *a, const void *b) {
    return ((Item *)b)->ratio - ((Item *)a)->ratio;
}

double fractionalKnapsack(Item items[], int n, int capacity) {
    // Sort items by value/weight ratio
    qsort(items, n, sizeof(Item), compare);

    double totalValue = 0.0;
    for (int i = 0; i < n; i++) {
        if (items[i].weight <= capacity) {
            // Take the full item
            totalValue += items[i].value;
            capacity -= items[i].weight;
        } else {
            // Take the fractional part of the item
            totalValue += items[i].value * ((double) capacity / items[i].weight);
            break;
        }
    }

    return totalValue;
}

int main() {
    Item items[] = {{10, 60, 0}, {20, 100, 0}, {30, 120, 0}};
    int n = sizeof(items) / sizeof(items[0]);
    int capacity = 50;
    
    // Calculate value per weight ratio for each item
    for (int i = 0; i < n; i++) {
        items[i].ratio = (double) items[i].value / items[i].weight;
    }
    
    double maxValue = fractionalKnapsack(items, n, capacity);
    printf("Maximum value in Knapsack = %.2f\n", maxValue);

    return 0;
}

Comparison Between Activity Selection and Fractional Knapsack Problem

PropertyActivity Selection ProblemFractional Knapsack Problem
Type of ProblemScheduling/Selection problemOptimization problem (maximize value)
Solution ApproachGreedy (Select the activity with the earliest finish time)Greedy (Select items based on value-to-weight ratio)
Items SelectionSelect activities with no overlapping time slotsSelect items fully or fractionally
Time ComplexityO(n log n) (Sorting + Greedy selection)O(n log n) (Sorting + Greedy selection)
Space ComplexityO(n)O(n)

সারসংক্ষেপ

  • Activity Selection Problem একটি Greedy Algorithm ব্যবহার করে সমাধান করা হয়, যেখানে একাধিক কর্মকাণ্ডের মধ্যে যেগুলি একে অপরের সাথে অতিক্রম করবে না, তা নির্বাচন করা হয়। এর Time Complexity O(n log n) হয়।
  • Fractional Knapsack Problem একটি Greedy Algorithm ব্যবহার করে সমাধান করা হয়, যেখানে আইটেমের মূল্য/ওজন অনুপাত অনুসারে আইটেমগুলি বাছাই করা হয়। এটি আংশিকভাবে আইটেম গ্রহণ করতে সক্ষম, যার ফলে এটি 0/1 Knapsack Problem থেকে আলাদা।

উভয় সমস্যা অপটিমাইজেশন পদ্ধতি ব্যবহার করে, তবে তাদের আবেদন ক্ষেত্র এবং অ্যালগরিদমের ধরণ ভিন্ন।

Content added By

Greedy এবং Dynamic Programming এর মধ্যে পার্থক্য

822

Greedy এবং Dynamic Programming এর মধ্যে পার্থক্য

Greedy (গ্রীডি) এবং Dynamic Programming (ডাইনামিক প্রোগ্রামিং) উভয়ই অপটিমাইজেশন সমস্যা সমাধানের কৌশল, তবে তাদের কাজ করার পদ্ধতি, কৌশল এবং প্রয়োগে মৌলিক পার্থক্য রয়েছে। উভয় কৌশল একাধিক উপ-সমস্যা বা সাব-অপটিমাল সমাধান ব্যবহার করে মূল সমস্যার সমাধান প্রদান করে, তবে তাদের মূল ধারণা এবং উপাদানগুলো ভিন্ন। নিচে উভয়ের মধ্যে কিছু মৌলিক পার্থক্য তুলে ধরা হলো:


১. মূল ধারণা (Core Concept)

  • Greedy Algorithm (গ্রীডি অ্যালগরিদম):
    • গ্রীডি অ্যালগরিদমটি প্রতি ধাপে সর্বোত্তম সিদ্ধান্ত নেয়, অর্থাৎ, এটি প্রতিটি ধাপে সর্বোচ্চ লাভ বা সবচেয়ে ভালো সিদ্ধান্ত গ্রহণ করে। এটি local optimal solution খোঁজার চেষ্টা করে এবং প্রতিটি পদক্ষেপের জন্য সর্বোত্তম সিদ্ধান্ত নেয়।
    • এটি global optimal solution (মোট সমস্যার সর্বোত্তম সমাধান) না পাওয়ার সম্ভাবনা থাকে, তবে অনেক সমস্যায় এটি কার্যকরী হতে পারে।
  • Dynamic Programming (ডাইনামিক প্রোগ্রামিং):
    • ডাইনামিক প্রোগ্রামিং অ্যালগরিদমটি optimal substructure এবং overlapping subproblems ধারণার ওপর কাজ করে। এটি ছোট ছোট উপ-সমস্যাগুলির সমাধান সংরক্ষণ করে এবং পরে সেগুলি পুনরায় ব্যবহার করে।
    • DP global optimal solution খুঁজে বের করার চেষ্টা করে, এবং এটি সব উপ-সমস্যার সঠিক সমাধান বের করে।

২. সমাধানের ধরণ (Solution Approach)

  • Greedy Algorithm:
    • প্রতিটি পদক্ষেপে local optimal সমাধান নেয়, এবং পরবর্তী পদক্ষেপে এই সিদ্ধান্তটি ভিত্তি করে কাজ করা হয়।
    • এটি কোনো backtracking বা future prediction করে না, শুধুমাত্র বর্তমান পরিস্থিতি অনুযায়ী সিদ্ধান্ত নেয়।
  • Dynamic Programming:
    • overlapping subproblems সমাধান করার জন্য memoization বা tabulation কৌশল ব্যবহার করে।
    • এটি পূর্ববর্তী পদক্ষেপের সঠিক সমাধান সংরক্ষণ করে এবং পরবর্তীতে এগুলি পুনরায় ব্যবহার করে। DP সবগুলো উপ-সমস্যার সমাধান একত্রিত করে পুরো সমস্যার সমাধান দেয়।

৩. মেমরি ব্যবহারের ধরন (Memory Usage)

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

৪. অপটিমাইজেশন সমস্যা সমাধান (Optimization Problem Solving)

  • Greedy Algorithm:
    • গ্রীডি অ্যালগরিদম সর্বদা local optimal solution খোঁজে, কিন্তু global optimal solution এর নিশ্চয়তা দেয় না।
    • Knapsack problem, Job Scheduling, Huffman coding ইত্যাদি গ্রীডি অ্যালগরিদমের মাধ্যমে সমাধান করা যেতে পারে, তবে সব ক্ষেত্রেই এটি সর্বোত্তম সমাধান নাও দিতে পারে।
  • Dynamic Programming:
    • ডাইনামিক প্রোগ্রামিং global optimal solution খোঁজে এবং এটি overlapping subproblems সমস্যাগুলির জন্য আদর্শ।
    • এটি Knapsack problem, Fibonacci sequence, Longest common subsequence ইত্যাদি অপটিমাইজেশন সমস্যার জন্য কার্যকরী, যেখানে উপ-সমস্যার সমাধান একাধিকবার ব্যবহৃত হয়।

৫. সময় এবং স্পেস জটিলতা (Time and Space Complexity)

  • Greedy Algorithm:
    • সাধারণত গ্রীডি অ্যালগরিদমের সময়ের জটিলতা অনেক কম থাকে এবং তা O(n log n) বা O(n) হতে পারে, কারণ এটি কেবলমাত্র একাধিক সিদ্ধান্ত নেয় এবং পুনরাবৃত্তি করে না।
    • স্পেস কমপ্লেক্সিটি সাধারণত কম থাকে, কারণ এটি শুধুমাত্র বর্তমান সিদ্ধান্ত সংরক্ষণ করে এবং অতিরিক্ত মেমরি ব্যবহার করে না।
  • Dynamic Programming:
    • ডাইনামিক প্রোগ্রামিং অ্যালগরিদমের সময় এবং স্পেস জটিলতা তুলনামূলকভাবে বেশি হতে পারে, যেমন O(n^2) বা O(nm), কারণ এটি উপ-সমস্যাগুলির জন্য ফলাফল সংরক্ষণ করে এবং সেই ফলাফলগুলো পুনরায় ব্যবহার করে।

৬. উদাহরণ (Examples)

  • Greedy Algorithm:
    • Huffman Coding: এটি গ্রীডি অ্যালগরিদম ব্যবহার করে বাইটের ছোট গড় আকার খোঁজে।
    • Fractional Knapsack Problem: গ্রীডি কৌশল দ্বারা পূর্ণ knapsack সমস্যা সমাধান করা যায়, তবে 0/1 Knapsack Problem তে গ্রীডি অ্যালগরিদম কার্যকরী নয়।
    • Activity Selection Problem: যে কোনো সময়ের মধ্যে সর্বাধিক কাজ নির্বাচন করার জন্য গ্রীডি অ্যালগরিদম ব্যবহৃত হয়।
  • Dynamic Programming:
    • Fibonacci Sequence: DP ব্যবহার করে পুনরাবৃত্তি সমাধানগুলো সংরক্ষণ করে দ্রুত ফলাফল পাওয়া যায়।
    • 0/1 Knapsack Problem: DP মাধ্যমে সর্বোত্তম সমাধান পাওয়া যায়, যেখানে একাধিক উপাদান এবং সীমাবদ্ধতা থাকে।
    • Longest Common Subsequence: দুটি স্ট্রিংয়ের মধ্যে সর্বাধিক সাধারণ উপসিকুয়েন্স খোঁজার জন্য DP ব্যবহার করা হয়।

৭. প্রয়োগ ক্ষেত্র (Application Areas)

  • Greedy Algorithm:
    • Job Scheduling (সময়সীমার মধ্যে কাজ সমাপ্ত করা),
    • Huffman Encoding (ডেটা কম্প্রেশন),
    • Graph Algorithms (কিছু MST অ্যালগরিদম যেমন Kruskal’s Algorithm, Prim’s Algorithm),
    • Activity Selection (অ্যাকটিভিটি নির্বাচন সমস্যায় সমাধান প্রদান)।
  • Dynamic Programming:
    • Shortest Path Problems (Dijkstra’s Algorithm, Floyd-Warshall Algorithm),
    • Knapsack Problem (0/1 Knapsack, Fractional Knapsack),
    • String Matching and Parsing (Longest Common Subsequence, Edit Distance),
    • Matrix Chain Multiplication (ম্যাট্রিক্স গুণফল অপ্টিমাইজেশন)।

সারসংক্ষেপ

বৈশিষ্ট্যGreedy AlgorithmDynamic Programming
মূল ধারণাLocal optimal solution each stepGlobal optimal solution through subproblem solutions
সিদ্ধান্ত গ্রহণপ্রতিটি পদক্ষেপে সর্বোত্তম সিদ্ধান্ত নেয়সকল উপ-সমস্যার সঠিক সমাধান নিয়ে গঠন করা হয়
সময়সীমাকম সময় জটিলতা, সাধারণত O(n) বা O(n log n)উচ্চ সময় জটিলতা, যেমন O(n^2) বা O(nm)
মেমরি ব্যবহারকম মেমরি ব্যবহার করেঅতিরিক্ত মেমরি ব্যবহৃত হয় (যেমন টেবিলিং বা মেমোইজেশন)
উদাহরণHuffman Coding, Activity Selection, Job SchedulingFibonacci Sequence, 0/1 Knapsack, Longest Common Subsequence

Greedy এবং Dynamic Programming উভয়ই কার্যকরী কৌশল, তবে তাদের ব্যবহার নির্ভর করে সমস্যা অনুসারে। Greedy Algorithm সাধারণত দ্রুত সমাধান প্রদান করে কিন্তু সর্বদা সেরা ফলাফল নাও দিতে পারে, তবে Dynamic Programming সবসময় সঠিক এবং সর্বোত্তম সমাধান প্রদান করে, যদিও এটি বেশি মেমরি এবং সময় নিতে পারে।

Content added By
Promotion

Are you sure to start over?

Loading...