Java Technologies Generics এর সাথে Collections Framework গাইড ও নোট

378

জাভার Generics এবং Collections Framework একসাথে মিলে টাইপ-সেফ, পুনঃব্যবহারযোগ্য এবং আরও কার্যকর ডেটা স্ট্রাকচার ব্যবহারের সুযোগ দেয়। Collections Framework-এ জেনেরিক্স যুক্ত করার ফলে কম্পাইল টাইমে টাইপ চেকিং সুনিশ্চিত হয়েছে এবং টাইপ কাস্টিংয়ের ঝামেলা কমেছে।


Generics Collections Framework এর ভূমিকা:

  1. Type Safety: নির্দিষ্ট ডেটা টাইপ ধরে রাখতে Collections Framework-এ Generics ব্যবহার হয়।
  2. Avoid Casting: রানটাইম টাইপ কাস্টিংয়ের প্রয়োজন নেই।
  3. Code Readability এবং Reusability: একই ধরনের কোড বারবার লেখার প্রয়োজন নেই।

Generics Collections Framework উদাহরণ:

১. List এর সাথে Generics:

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

public class Main {
    public static void main(String[] args) {
        // String টাইপের জন্য একটি List
        List<String> names = new ArrayList<>();
        names.add("Alice");
        names.add("Bob");

        // কম্পাইল টাইম ত্রুটি
        // names.add(123);

        for (String name : names) {
            System.out.println(name);
        }
    }
}

আউটপুট:

Alice
Bob

২. Map এর সাথে Generics:

import java.util.HashMap;
import java.util.Map;

public class Main {
    public static void main(String[] args) {
        // Key: String, Value: Integer
        Map<String, Integer> ageMap = new HashMap<>();
        ageMap.put("Alice", 25);
        ageMap.put("Bob", 30);

        // কম্পাইল টাইম ত্রুটি
        // ageMap.put(123, "Invalid");

        for (Map.Entry<String, Integer> entry : ageMap.entrySet()) {
            System.out.println(entry.getKey() + " is " + entry.getValue() + " years old.");
        }
    }
}

আউটপুট:

Alice is 25 years old.
Bob is 30 years old.

৩. Set এর সাথে Generics:

import java.util.HashSet;
import java.util.Set;

public class Main {
    public static void main(String[] args) {
        // Integer টাইপের জন্য একটি Set
        Set<Integer> numbers = new HashSet<>();
        numbers.add(10);
        numbers.add(20);
        numbers.add(30);

        for (Integer number : numbers) {
            System.out.println(number);
        }
    }
}

আউটপুট:

10
20
30

Generics Collections এর গুরুত্বপূর্ণ বৈশিষ্ট্য:

১. টাইপ সেফ Collections:

Generics Collections টাইপ সেফ। তাই ভুল ডেটা টাইপ যুক্ত করা যায় না।

List<Integer> numbers = new ArrayList<>();
numbers.add(10);
// numbers.add("String"); // কম্পাইল টাইম ত্রুটি

২. টাইপ কাস্টিংয়ের প্রয়োজন নেই:

Generics Collections ব্যবহার করলে টাইপ কাস্টিংয়ের প্রয়োজন হয় না।

// Without Generics
List list = new ArrayList();
list.add("Hello");
String message = (String) list.get(0); // Casting required

// With Generics
List<String> listWithGenerics = new ArrayList<>();
listWithGenerics.add("Hello");
String messageWithGenerics = listWithGenerics.get(0); // No casting needed

৩. Wildcards (?) এর ব্যবহার:

Wildcard বিভিন্ন ধরনের ডেটা হ্যান্ডেল করতে সাহায্য করে।

  • ? extends T: T বা T এর সাবক্লাস গ্রহণ করে।
  • ? super T: T বা T এর সুপারক্লাস গ্রহণ করে।
  • ?: যেকোনো টাইপ গ্রহণ করে।
import java.util.List;

public class Main {
    public static void printList(List<?> list) {
        for (Object obj : list) {
            System.out.println(obj);
        }
    }

    public static void main(String[] args) {
        List<String> stringList = List.of("A", "B", "C");
        List<Integer> intList = List.of(1, 2, 3);

        printList(stringList);
        printList(intList);
    }
}

আউটপুট:

A
B
C
1
2
3

Collections Framework এর Generics শ্রেণিবিন্যাস:

১. List (Ordered Collection):

  • ArrayList<E>
  • LinkedList<E>
  • Vector<E>

২. Set (Unique Elements):

  • HashSet<E>
  • LinkedHashSet<E>
  • TreeSet<E>

৩. Map (Key-Value Pairs):

  • HashMap<K, V>
  • TreeMap<K, V>
  • LinkedHashMap<K, V>

৪. Queue/Deque:

  • PriorityQueue<E>
  • ArrayDeque<E>

Generics এর সীমাবদ্ধতা:

  1. Primitive Types: Generics সরাসরি প্রিমিটিভ টাইপ ব্যবহার করতে পারে না। তবে Integer, Double ইত্যাদি Autoboxing এর মাধ্যমে ব্যবহার করা যায়।

    List<int> list = new ArrayList<>(); // ত্রুটি
    List<Integer> list = new ArrayList<>(); // সঠিক
    
  2. Runtime Type Erasure: Generics কম্পাইল টাইমে কাজ করে; রানটাইমে Generics এর টাইপ মুছে যায়।

Generics এবং Collections Framework একসাথে ব্যবহারে টাইপ-সেফ এবং আরও কার্যকর প্রোগ্রাম লেখা সম্ভব হয়। এটি শুধু কম্পাইল টাইম ত্রুটি কমায় না, বরং কোডের রিডেবিলিটি এবং রিইউজেবিলিটি বাড়ায়।

Content added By

Java Collections Framework এবং Generics এর সম্পর্ক

295

জাভা জেনেরিক্স (Java Generics) এবং Java Collections Framework একে অপরের সাথে ঘনিষ্ঠভাবে সম্পর্কিত। Java Collections Framework জাভার এক সেট ক্লাস এবং ইন্টারফেস সরবরাহ করে যা ডেটা স্টোর এবং ম্যানেজ করার জন্য ব্যবহৃত হয়। Generics Collections Framework-কে আরও শক্তিশালী, নিরাপদ এবং কার্যকর করে তোলে।


Java Collections Framework এবং Generics এর ভূমিকা

  1. Generics এর আগে Collection Framework
    Generics ব্যবহার করার আগে, Collections Framework-এ ডেটা ইনসার্ট এবং রিট্রাইভ করার সময় টাইপ কাস্টিং করতে হতো। এটি runtime errors-এর ঝুঁকি বাড়াতো।
// Without Generics
import java.util.*;

public class Main {
    public static void main(String[] args) {
        List list = new ArrayList(); // Non-generic List
        list.add("Hello");
        list.add(10); // Allowed, no compile-time error

        for (Object obj : list) {
            String str = (String) obj; // Runtime error for non-String objects
            System.out.println(str);
        }
    }
}

সমস্যা:

  • Compile-time টাইপ চেকিং অনুপস্থিত।
  • Runtime টাইপ কাস্টিং ব্যর্থ হতে পারে এবং ClassCastException ঘটতে পারে।

  1. Generics এর সাথে Collection Framework
    Generics ব্যবহার করে, Collection Framework টাইপ সেফটি নিশ্চিত করে এবং টাইপ কাস্টিংয়ের প্রয়োজনীয়তা দূর করে।
// With Generics
import java.util.*;

public class Main {
    public static void main(String[] args) {
        List<String> list = new ArrayList<>(); // Generic List
        list.add("Hello");
        // list.add(10); // Compile-time error

        for (String str : list) {
            System.out.println(str); // No type casting needed
        }
    }
}

সুবিধা:

  • Compile-time টাইপ চেকিং নিশ্চিত হয়।
  • কোড নিরাপদ ও রিডেবল হয়।

Generics এবং Collections Framework-এর সম্পর্কিত উদাহরণ

1. Generics ব্যবহার করে List

import java.util.*;

public class Main {
    public static void main(String[] args) {
        List<Integer> numbers = new ArrayList<>();
        numbers.add(1);
        numbers.add(2);
        numbers.add(3);

        for (Integer num : numbers) {
            System.out.println(num);
        }
    }
}

2. Generics ব্যবহার করে Map

import java.util.*;

public class Main {
    public static void main(String[] args) {
        Map<String, Integer> map = new HashMap<>();
        map.put("One", 1);
        map.put("Two", 2);

        for (Map.Entry<String, Integer> entry : map.entrySet()) {
            System.out.println(entry.getKey() + " = " + entry.getValue());
        }
    }
}

Common Collection Interfaces এবং Generics

  1. List
    • ArrayList<E>, LinkedList<E> ইত্যাদি।
    • উদাহরণ: List<String> list = new ArrayList<>();
  2. Set
    • HashSet<E>, TreeSet<E> ইত্যাদি।
    • উদাহরণ: Set<Integer> set = new HashSet<>();
  3. Map<K, V>
    • HashMap<K, V>, TreeMap<K, V> ইত্যাদি।
    • উদাহরণ: Map<String, Integer> map = new HashMap<>();
  4. Queue
    • PriorityQueue<E>, LinkedList<E> ইত্যাদি।
    • উদাহরণ: Queue<String> queue = new LinkedList<>();
  5. Deque
    • ArrayDeque<E>, LinkedList<E> ইত্যাদি।
    • উদাহরণ: Deque<Double> deque = new ArrayDeque<>();

Generics Collections-এর সুবিধা

  1. Type Safety নিশ্চিত করে
    Collection Framework-এ যে টাইপ ডেটা যোগ করা হবে, Generics সেটি compile-time এ চেক করে। ভুল টাইপ অ্যাড করলে compile-time error দেখায়।
  2. Runtime Errors প্রতিরোধ করে
    Generics টাইপ সংক্রান্ত runtime errors (যেমন ClassCastException) প্রতিরোধ করে।
  3. কোড রিডেবল এবং ম্যানেজযোগ্য করে
    Generics-এর কারণে টাইপ কাস্টিংয়ের প্রয়োজন হয় না, যা কোডকে আরও পরিষ্কার ও সহজ করে তোলে।

Wildcard Generics এবং Collections

Generics-এর সাথে wildcard (?) ব্যবহার করে বিভিন্ন টাইপ হ্যান্ডেল করা যায়।

উদাহরণ:

import java.util.*;

public class Main {
    public static void printList(List<?> list) {
        for (Object obj : list) {
            System.out.println(obj);
        }
    }

    public static void main(String[] args) {
        List<Integer> intList = List.of(1, 2, 3);
        List<String> strList = List.of("A", "B", "C");

        printList(intList); // Works with any type
        printList(strList);
    }
}

Bounded Wildcards

  1. Upper Bound (<? extends T>)
    এটি নির্ধারণ করে যে Generics টাইপ হবে T বা তার সাবটাইপ।

    public static void printNumbers(List<? extends Number> list) {
        for (Number num : list) {
            System.out.println(num);
        }
    }
    
  2. Lower Bound (<? super T>)
    এটি নির্ধারণ করে যে Generics টাইপ হবে T বা তার সুপারটাইপ।

    public static void addNumbers(List<? super Integer> list) {
        list.add(10); // Allowed
    }
    

  • Java Collections Framework এবং Generics একসাথে টাইপ সেফটি এবং কোড রিডেবিলিটি নিশ্চিত করে।
  • Generics ব্যবহার করলে compile-time টাইপ চেকিং সম্ভব হয়, যা runtime errors কমায়।
  • Collection Framework-এর সাথে Generics ব্যবহারের মাধ্যমে ডেটা ম্যানিপুলেশন সহজ, কার্যকর, এবং নিরাপদ হয়।
Content added By

List, Set, এবং Map এর সাথে Generics ব্যবহার

301

জাভা জেনেরিক্স (Java Generics) List, Set, এবং Map এর মতো Collection Framework এর ক্লাস ও ইন্টারফেসে ব্যাপকভাবে ব্যবহৃত হয়। Generics Collections আমাদের টাইপ সেফটি প্রদান করে এবং ডাউনকাস্টিং এড়ায়।


List এর সাথে Generics

List ইন্টারফেস একটি অর্ডারড কালেকশন যা ডুপ্লিকেট আইটেম সাপোর্ট করে। Generics ব্যবহার করলে আমরা নিশ্চিত করতে পারি যে লিস্ট একটি নির্দিষ্ট টাইপের অবজেক্ট গ্রহণ করবে।

উদাহরণ:

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

public class Main {
    public static void main(String[] args) {
        List<String> stringList = new ArrayList<>();
        stringList.add("Java");
        stringList.add("Generics");
        stringList.add("Example");

        for (String item : stringList) {
            System.out.println(item);
        }
    }
}

আউটপুট:

Java
Generics
Example

বিঃদ্রঃ Generics ব্যবহারের ফলে টাইপ কাস্টিং প্রয়োজন হয় না।


Set এর সাথে Generics

Set একটি আনঅর্ডারড কালেকশন যা ডুপ্লিকেট আইটেম গ্রহণ করে না। Generics ব্যবহার করলে আমরা টাইপ সেফটি নিশ্চিত করতে পারি।

উদাহরণ:

import java.util.HashSet;
import java.util.Set;

public class Main {
    public static void main(String[] args) {
        Set<Integer> numberSet = new HashSet<>();
        numberSet.add(10);
        numberSet.add(20);
        numberSet.add(30);

        for (Integer num : numberSet) {
            System.out.println(num);
        }
    }
}

আউটপুট:

10
20
30

বিঃদ্রঃ এখানে আইটেমগুলো আনঅর্ডারড অবস্থায় প্রিন্ট হতে পারে।


Map এর সাথে Generics

Map একটি key-value পেয়ার কালেকশন, যেখানে প্রতিটি key ইউনিক হয়। Generics ব্যবহার করে আমরা key এবং value উভয়ের টাইপ নির্ধারণ করতে পারি।

উদাহরণ:

import java.util.HashMap;
import java.util.Map;

public class Main {
    public static void main(String[] args) {
        Map<Integer, String> studentMap = new HashMap<>();
        studentMap.put(1, "John");
        studentMap.put(2, "Emma");
        studentMap.put(3, "Alex");

        for (Map.Entry<Integer, String> entry : studentMap.entrySet()) {
            System.out.println("ID: " + entry.getKey() + ", Name: " + entry.getValue());
        }
    }
}

আউটপুট:

ID: 1, Name: John
ID: 2, Name: Emma
ID: 3, Name: Alex

Generics এর মাধ্যমে:

  1. টাইপ সেফটি: টাইপ নির্ধারণ করে কম্পাইল টাইমে ত্রুটি সনাক্ত করা যায়।
  2. কোড সরলতা: টাইপ কাস্টিং এর ঝামেলা এড়ানো যায়।
  3. ডেটা ম্যানেজমেন্ট: টাইপ নির্দিষ্ট করে কাজ সহজ হয়।

List, Set, এবং Map এর সাথে Generics ব্যবহার করলে ডেটা ম্যানিপুলেশন আরও বেশি কার্যকর ও নিরাপদ হয়।

Content added By

Iterators এর মাধ্যমে Type-Safe Iteration

334

জাভা জেনেরিক্স (Java Generics) ব্যবহার করে Iterators এর মাধ্যমে টাইপ-সেফ ইটারেশন (Type-Safe Iteration) অর্জন করা সম্ভব। এটি Collections Framework এ জেনেরিক্স প্রবর্তনের অন্যতম প্রধান সুবিধা। টাইপ-সেফ ইটারেশন নিশ্চিত করে যে, ইটারেটর শুধুমাত্র নির্দিষ্ট টাইপের উপাদানগুলি অ্যাক্সেস করবে এবং অন্য কোনো টাইপের ডেটা কম্পাইল টাইমেই ত্রুটি সৃষ্টি করবে।


Iterators এবং Generics-এর ব্যবহার

উদাহরণ: টাইপ-সেফ ইটারেশন

import java.util.ArrayList;
import java.util.Iterator;

public class Main {
    public static void main(String[] args) {
        // String টাইপের জন্য একটি ArrayList
        ArrayList<String> list = new ArrayList<>();
        
        // উপাদান যোগ করা
        list.add("Java");
        list.add("Generics");
        list.add("Iterator");

        // Iterator ব্যবহার করে টাইপ-সেফ ইটারেশন
        Iterator<String> iterator = list.iterator();
        while (iterator.hasNext()) {
            String element = iterator.next();
            System.out.println(element);
        }
    }
}

আউটপুট:

Java
Generics
Iterator

টাইপ-সেফ ইটারেশন এর সুবিধা

  1. Compile-Time Type Checking:
    • জেনেরিক্স ব্যবহারে, ইটারেটর শুধুমাত্র নির্দিষ্ট টাইপের ডেটা অ্যাক্সেস করতে পারে।
    • ভুল টাইপের ডেটা অ্যাক্সেস করার চেষ্টা করলে, এটি কম্পাইল টাইমেই ত্রুটি প্রদর্শন করবে।
  2. Type Casting এর প্রয়োজন নেই:
    • জেনেরিক্স ব্যবহারে টাইপকাস্টিংয়ের ঝামেলা দূর হয়।
    • পূর্বে Object থেকে নির্দিষ্ট টাইপে কাস্টিং করতে হতো, যা রানটাইমে ক্লাসকাস্ট এক্সসেপশন (ClassCastException) সৃষ্টি করতে পারত।

Generics ছাড়া Iterators-এর উদাহরণ

জেনেরিক্স ছাড়াই ইটারেটর ব্যবহারের উদাহরণ (Java 1.4 এবং এর পূর্ববর্তী):

import java.util.ArrayList;
import java.util.Iterator;

public class Main {
    public static void main(String[] args) {
        // Non-generic ArrayList
        ArrayList list = new ArrayList();
        
        // উপাদান যোগ করা
        list.add("Java");
        list.add(100); // ভুল টাইপ যোগ করা সম্ভব

        // Iterator ব্যবহার
        Iterator iterator = list.iterator();
        while (iterator.hasNext()) {
            // টাইপ কাস্টিং প্রয়োজন
            String element = (String) iterator.next(); // রানটাইম এক্সসেপশন হতে পারে
            System.out.println(element);
        }
    }
}

সমস্যাগুলি:

  1. ClassCastException:
    • ভুল টাইপের ডেটা কাস্ট করার সময় রানটাইম এক্সসেপশন হতে পারে।
  2. Code Readability কমে যায়:
    • প্রতিটি উপাদানের জন্য টাইপ কাস্ট করতে হয়, যা কোডকে জটিল করে তোলে।

Generics ব্যবহার করলে Type-Safe Iteration নিশ্চিত হয়

জেনেরিক্স ব্যবহার করে টাইপ-সেফ:

ArrayList<String> list = new ArrayList<>();
list.add("Java");
// list.add(100); // কম্পাইল টাইম ত্রুটি

জেনেরিক্স ব্যবহার করে টাইপ-সেফ ইটারেশন জাভার কোডের মান উন্নত করে এবং ClassCastException এর ঝুঁকি দূর করে। এটি Collections Framework এর প্রধান একটি বৈশিষ্ট্য, যা ডেভেলপারদের কোড লেখার সময় আরও নির্ভুলতা ও নিরাপত্তা প্রদান করে।

Content added By

Collections এর জন্য Wildcards এবং Bounded Types এর প্রয়োগ

303

জাভায় জেনেরিক্স ব্যবহার করার সময় Wildcards এবং Bounded Types দুটি গুরুত্বপূর্ণ ধারণা। এগুলো Collections Framework-এর সাথে ব্যবহার করে টাইপ-সেইফ, ফ্লেক্সিবল এবং পুনঃব্যবহারযোগ্য কোড তৈরি করা যায়।


Wildcards:

Wildcards (?) ব্যবহার করে এমন জেনেরিক টাইপগুলো তৈরি করা যায় যেগুলো অজানা বা পরিবর্তনশীল হতে পারে।

Wildcards এর ধরণ:

  1. Unbounded Wildcard (?): এটি ব্যবহার করা হয় যখন কোনো নির্দিষ্ট টাইপের প্রতি সীমাবদ্ধতা নেই।

    public void printList(List<?> list) {
        for (Object obj : list) {
            System.out.println(obj);
        }
    }
    

    উদাহরণ:

    List<String> stringList = Arrays.asList("Java", "Generics", "Wildcard");
    List<Integer> intList = Arrays.asList(10, 20, 30);
    
    printList(stringList); // Output: Java, Generics, Wildcard
    printList(intList);    // Output: 10, 20, 30
    
  2. Upper Bounded Wildcard (<? extends Type>): এটি ব্যবহার করা হয় যখন একটি টাইপ নির্দিষ্ট ক্লাস বা তার সাবক্লাস হতে পারে।

    public void printNumbers(List<? extends Number> list) {
        for (Number num : list) {
            System.out.println(num);
        }
    }
    

    উদাহরণ:

    List<Integer> integers = Arrays.asList(1, 2, 3);
    List<Double> doubles = Arrays.asList(1.1, 2.2, 3.3);
    
    printNumbers(integers); // Output: 1, 2, 3
    printNumbers(doubles);  // Output: 1.1, 2.2, 3.3
    
  3. Lower Bounded Wildcard (<? super Type>): এটি ব্যবহার করা হয় যখন একটি টাইপ নির্দিষ্ট ক্লাস বা তার সুপারক্লাস হতে পারে।

    public void addNumbers(List<? super Integer> list) {
        list.add(10);
        list.add(20);
    }
    

    উদাহরণ:

    List<Number> numbers = new ArrayList<>();
    addNumbers(numbers);
    
    System.out.println(numbers); // Output: [10, 20]
    

Bounded Types:

Bounded Types ব্যবহার করা হয় যখন একটি জেনেরিক টাইপকে নির্দিষ্ট টাইপের মধ্যে সীমাবদ্ধ করতে হয়।

Upper Bound (<T extends Type>)

public <T extends Number> double calculateSum(List<T> list) {
    double sum = 0.0;
    for (T num : list) {
        sum += num.doubleValue();
    }
    return sum;
}

উদাহরণ:

List<Integer> integers = Arrays.asList(1, 2, 3, 4);
List<Double> doubles = Arrays.asList(1.1, 2.2, 3.3);

System.out.println(calculateSum(integers)); // Output: 10.0
System.out.println(calculateSum(doubles));  // Output: 6.6

Multiple Bounds (<T extends Class & Interface>):

public <T extends Number & Comparable<T>> T findMax(T a, T b) {
    return a.compareTo(b) > 0 ? a : b;
}

উদাহরণ:

System.out.println(findMax(10, 20));   // Output: 20
System.out.println(findMax(3.14, 2.71)); // Output: 3.14

Wildcards vs Bounded Types:

AspectWildcards (?)Bounded Types (<T extends Type>)
UsageFlexible and used in method argumentsUsed in method signatures and class definitions
Type InformationWorks with unknown typesWorks with specific bounded types
Common Use CasesReading data from collectionsPerforming operations on specific types

উদাহরণ: Collections-এ Wildcards এবং Bounded Types

import java.util.*;

public class WildcardAndBoundedTypesExample {
    // Upper Bounded Wildcard Example
    public static void displayNumbers(List<? extends Number> list) {
        for (Number num : list) {
            System.out.println(num);
        }
    }

    // Lower Bounded Wildcard Example
    public static void addElements(List<? super Integer> list) {
        list.add(1);
        list.add(2);
        list.add(3);
    }

    public static void main(String[] args) {
        List<Integer> integers = new ArrayList<>(Arrays.asList(10, 20, 30));
        List<Number> numbers = new ArrayList<>();
        
        displayNumbers(integers); // Output: 10, 20, 30
        addElements(numbers);
        
        System.out.println(numbers); // Output: [1, 2, 3]
    }
}

Wildcards এবং Bounded Types জেনেরিক্সে ফ্লেক্সিবিলিটি এবং টাইপ সেফটি প্রদান করে। Collections Framework-এ তাদের ব্যবহারে কোড আরো রিডেবল, পুনঃব্যবহারযোগ্য এবং কার্যকরী হয়।

Content added By
Promotion

Are you sure to start over?

Loading...