Atomics এর বিকল্প সমাধান: ReadWriteLocks, Synchronized Collections, ReentrantLock, ইত্যাদি

Atomics এর Alternatives - অ্যাটমিক্স (Atomics) - Web Development

262

Atomics API হল একাধিক থ্রেডে শেয়ার করা মেমোরি ব্যবস্থাপনার জন্য একটি শক্তিশালী টুল। তবে, এটি সব পরিস্থিতিতে উপযুক্ত নাও হতে পারে, এবং কিছু সীমাবদ্ধতা বা চ্যালেঞ্জের সম্মুখীন হতে পারে, যেমন deadlock, complexity, এবং performance issues। এই কারণে, অন্যান্য synchronization এবং locking mechanisms যেমন ReadWriteLocks, Synchronized Collections, ReentrantLock ইত্যাদি ব্যবহার করা হতে পারে।

এই টুলগুলো সাধারণত thread safety এবং data consistency নিশ্চিত করতে ব্যবহৃত হয়, বিশেষ করে মাল্টি-থ্রেডেড প্রোগ্রামিংয়ে। চলুন, এগুলোর সম্পর্কে বিস্তারিত আলোচনা করি।


1. ReadWriteLocks

ReadWriteLocks একটি লক মেকানিজম যা readers এবং writers এর মধ্যে পার্থক্য করে। যখন অনেক থ্রেড একই ডেটাকে পড়তে চায়, তখন একাধিক থ্রেড একসাথে ডেটা পড়তে পারবে (read lock)। তবে, যদি কোনো থ্রেড ডেটা পরিবর্তন করতে চায় (write lock), তখন সেই থ্রেড এককভাবে ডেটা পরিবর্তন করতে পারে এবং অন্য কোনো থ্রেড তখন পড়তে বা লিখতে পারবে না।

বৈশিষ্ট্য:

  • Read lock: একাধিক থ্রেড একই সময় ডেটা পড়তে পারবে।
  • Write lock: যখন লেখার প্রয়োজন হয়, তখন অন্যান্য থ্রেডের জন্য লেখার বা পড়ার অনুমতি বন্ধ থাকবে।

ব্যবহার:

এটি মূলত যখন ডেটার ওপর প্রচুর রিড (read) অপারেশন হয় এবং লেখার (write) অপারেশন খুব কম হয়, তখন কার্যকর।

উদাহরণ (Java):

import java.util.concurrent.locks.ReadWriteLock;
import java.util.concurrent.locks.ReentrantReadWriteLock;

public class ReadWriteLockExample {
    private final ReadWriteLock lock = new ReentrantReadWriteLock();
    private int sharedData = 0;

    public void read() {
        lock.readLock().lock(); // Read lock acquired
        try {
            System.out.println("Reading data: " + sharedData);
        } finally {
            lock.readLock().unlock();
        }
    }

    public void write(int newValue) {
        lock.writeLock().lock(); // Write lock acquired
        try {
            sharedData = newValue;
            System.out.println("Writing data: " + sharedData);
        } finally {
            lock.writeLock().unlock();
        }
    }
}

2. Synchronized Collections

Synchronized Collections হল Java এর Collections Framework এর অংশ, যেখানে synchronized কন্টেইনার বা thread-safe কোলেকশনের সুবিধা পাওয়া যায়। সিঙ্ক্রোনাইজেশন নিশ্চিত করার জন্য সাধারণত ডেটা কন্টেইনারগুলোর ওপর লক ব্যবহার করা হয়, যা মাল্টি-থ্রেডেড পরিবেশে সঠিকভাবে কাজ করতে সাহায্য করে।

বৈশিষ্ট্য:

  • Thread-safe: এটি thread-safe collections তৈরি করতে সাহায্য করে যেমন List, Set, Map ইত্যাদি।
  • Automatic synchronization: ডেটা সংরক্ষণের সময় থ্রেড নিরাপত্তা নিশ্চিত করা হয়।

ব্যবহার:

এটি সাধারণত ছোট অ্যাপ্লিকেশনের জন্য ব্যবহৃত হয় যেখানে শেয়ার করা কন্টেইনারের ওপর একাধিক থ্রেড কাজ করবে।

উদাহরণ (Java):

import java.util.Collections;
import java.util.List;
import java.util.ArrayList;

public class SynchronizedCollectionExample {
    public static void main(String[] args) {
        List<Integer> synchronizedList = Collections.synchronizedList(new ArrayList<>());

        // Synchronized write
        synchronizedList.add(1);
        synchronizedList.add(2);

        // Synchronized read
        synchronized (synchronizedList) {
            for (int num : synchronizedList) {
                System.out.println(num);
            }
        }
    }
}

3. ReentrantLock

ReentrantLock একটি Java এর Lock ক্লাস যা explicit locking প্রদান করে, এবং এটি থ্রেডকে একাধিকবার একটি লক অর্জন করার অনুমতি দেয়, তাই এটি "reentrant" বলা হয়। এটি fair এবং non-fair লকিং সমর্থন করে, এবং নির্দিষ্ট থ্রেডের জন্য লক অনুক্রম নির্ধারণ করতে পারে।

বৈশিষ্ট্য:

  • Reentrant: একটি থ্রেড একাধিকবার একই লক অর্জন করতে পারে।
  • Fair Lock: এটি নিশ্চিত করে যে প্রতিটি থ্রেড একে অপরের আগে লক পাবে, যদি fair mode সিলেক্ট করা হয়।
  • Condition Support: Condition ক্লাসের মাধ্যমে থ্রেডের মধ্যে সিঙ্ক্রোনাইজেশন পরিচালনা করা যেতে পারে।

ব্যবহার:

এটি তখন ব্যবহৃত হয় যখন থ্রেডগুলির মধ্যে explicit lock management এবং fine-grained control প্রয়োজন।

উদাহরণ (Java):

import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;

public class ReentrantLockExample {
    private final Lock lock = new ReentrantLock();
    private int sharedData = 0;

    public void increment() {
        lock.lock(); // Lock acquired
        try {
            sharedData++;
            System.out.println("Incremented data: " + sharedData);
        } finally {
            lock.unlock(); // Lock released
        }
    }
}

4. Semaphore

Semaphore একটি synchronization primitive যা resource counting এর জন্য ব্যবহৃত হয়। এটি থ্রেডের সংখ্যা সীমাবদ্ধ করে এবং এটি সাধারণত resource pool বা rate limiting এর জন্য ব্যবহৃত হয়।

বৈশিষ্ট্য:

  • Count-based locking: একটি নির্দিষ্ট সংখ্যক থ্রেডকে একযোগে critical section এ প্রবেশ করতে দেয়।
  • Resource management: যখন একাধিক থ্রেড একই রিসোর্স ব্যবহার করতে চায়, তখন এটি ব্যবহৃত হয়।

ব্যবহার:

এটি তখন ব্যবহৃত হয় যখন আপনাকে একাধিক থ্রেডকে নির্দিষ্ট পরিমাণ রিসোর্স ব্যবহার করার অনুমতি দিতে হয়।

উদাহরণ (Java):

import java.util.concurrent.Semaphore;

public class SemaphoreExample {
    private final Semaphore semaphore = new Semaphore(2); // 2 permits

    public void accessResource() {
        try {
            semaphore.acquire(); // Acquire permit
            System.out.println("Resource accessed by " + Thread.currentThread().getName());
        } catch (InterruptedException e) {
            Thread.currentThread().interrupt();
        } finally {
            semaphore.release(); // Release permit
        }
    }
}

5. Latch and CountDownLatch

Latch এবং CountDownLatch ক্লাসগুলো একাধিক থ্রেডের সমন্বয় তৈরি করার জন্য ব্যবহৃত হয়। বিশেষ করে এটি তখন ব্যবহার হয় যখন আপনাকে একাধিক থ্রেডের মধ্যে সিঙ্ক্রোনাইজেশন নিশ্চিত করতে হয়, যেমন থ্রেডগুলো কাজ শেষ করার পরে কোন থ্রেড প্রধান কাজ করতে পারে।

ব্যবহার:

এটি ব্যবহৃত হয় যখন একটি থ্রেড বা গ্রুপ থ্রেড অন্য থ্রেডগুলোর কাজ শেষ হওয়ার জন্য অপেক্ষা করে।

উদাহরণ (Java):

import java.util.concurrent.CountDownLatch;

public class CountDownLatchExample {
    public static void main(String[] args) throws InterruptedException {
        CountDownLatch latch = new CountDownLatch(3);

        Runnable task = () -> {
            System.out.println(Thread.currentThread().getName() + " working");
            latch.countDown(); // Decrease latch count
        };

        Thread t1 = new Thread(task);
        Thread t2 = new Thread(task);
        Thread t3 = new Thread(task);

        t1.start();
        t2.start();
        t3.start();

        latch.await(); // Wait until latch count is zero
        System.out.println("All tasks completed, proceeding further.");
    }
}

উপসংহার

Atomics এর বিকল্প সমাধান হিসেবে ReadWriteLocks, Synchronized Collections, ReentrantLock, Semaphore, CountDownLatch ইত্যাদি একাধিক synchronization mechanisms রয়েছে। এগুলোর প্রতিটির নিজস্ব সুবিধা এবং ব্যবহার রয়েছে, যা বিভিন্ন প্রোগ্রামিং পরিস্থিতিতে প্রয়োগ করা যেতে পারে। আপনার প্রয়োজনের উপর নির্ভর করে, সঠিক locking এবং synchronization কৌশল নির্বাচন করা অত্যন্ত গুরুত্বপূর্ণ।

Content added By
Promotion

Are you sure to start over?

Loading...