DEV Community

Anh Trần Tuấn
Anh Trần Tuấn

Posted on • Originally published at tuanh.net on

Mastering Java Generics: A Comprehensive Guide with Code Examples

1. Understanding the Basics of Java Generics

Image

Java Generics allow you to define classes, interfaces, and methods with type parameters. This means you can write code that works with different data types while ensuring type safety at compile time.

1.1 What Are Generics?

Generics are a feature introduced in Java 5 that allow you to define a class, interface, or method with a placeholder for the data type it operates on. This helps to create classes and methods that can work with any type, while still providing compile-time type safety.

Example:

public class Box<T> {
    private T value;

    public void setValue(T value) {
        this.value = value;
    }

    public T getValue() {
        return value;
    }
}
Enter fullscreen mode Exit fullscreen mode

In this example, T is a type parameter that can be replaced with any data type when you create an instance of Box.

1.2 Why Use Generics?

Generics provide several benefits:

  • Type Safety : They eliminate the need for casting and reduce runtime errors by catching type mismatches at compile time.
  • Code Reusability : They allow you to create a single class or method that can handle different data types.
  • Eliminates Casting : Using generics reduces the need for type casting, making your code cleaner and less error-prone.

Example:

Without Generics:

List list = new ArrayList();
list.add("Hello");
String s = (String) list.get(0); // Casting needed
Enter fullscreen mode Exit fullscreen mode

With Generics:

List<String> list = new ArrayList<>();
list.add("Hello");
String s = list.get(0); // No casting needed
Enter fullscreen mode Exit fullscreen mode

2. Working with Generics in Classes

Generics can be used with classes to ensure type safety and flexibility. Here's how you can effectively use generics in class definitions.

2.1 Generic Class Example

Consider a generic class Pair that holds two values of possibly different types:

public class Pair<K, V> {
    private K key;
    private V value;

    public Pair(K key, V value) {
        this.key = key;
        this.value = value;
    }

    public K getKey() {
        return key;
    }

    public V getValue() {
        return value;
    }
}
Enter fullscreen mode Exit fullscreen mode

In this example, K and V are type parameters that can be specified when creating an instance of Pair.

Usage Example:

Pair<String, Integer> pair = new Pair<>("Age", 30);
System.out.println("Key: " + pair.getKey() + ", Value: " + pair.getValue());
Enter fullscreen mode Exit fullscreen mode

2.2 Bounded Type Parameters

Sometimes, you might want to restrict the types that can be used with a generic class or method. Bounded type parameters allow you to specify a range of acceptable types.

Example:

public class NumberUtils {
    public static <T extends Number> void printNumber(T number) {
        System.out.println("Number: " + number);
    }
}
Enter fullscreen mode Exit fullscreen mode

Here, T is restricted to subclasses of Number, such as Integer, Double, etc.

Usage Example:

NumberUtils.printNumber(10); // Valid
NumberUtils.printNumber(10.5); // Valid
Enter fullscreen mode Exit fullscreen mode

3. Generics in Methods

Generics are not limited to classes. They can also be used in methods to create flexible and type-safe methods.

3.1 Generic Method Example

Here's a method that returns the maximum of two values:

public class Util {
    public static <T extends Comparable<T>> T max(T a, T b) {
        return a.compareTo(b) > 0 ? a : b;
    }
}
Enter fullscreen mode Exit fullscreen mode

In this method, T is a type parameter that must extend Comparable, ensuring that T can be compared.

Usage Example:

System.out.println(Util.max(5, 10)); // Output: 10
System.out.println(Util.max("apple", "banana")); // Output: banana
Enter fullscreen mode Exit fullscreen mode

3.2 Wildcards in Generics

Wildcards allow you to specify a range of acceptable types in a more flexible manner. For example, ? extends T denotes an unknown type that extends T, while ? super T denotes an unknown type that is a supertype of T.

Example:

public void printList(List<? extends Number> list) {
    for (Number number : list) {
        System.out.println(number);
    }
}
Enter fullscreen mode Exit fullscreen mode

Usage Example:

List<Integer> integers = Arrays.asList(1, 2, 3);
printList(integers); // Prints the list of integers
Enter fullscreen mode Exit fullscreen mode

4. Conclusion

Mastering Java Generics can greatly enhance the flexibility and safety of your code. By understanding how to use generics in classes, methods, and with wildcards, you'll be able to write more robust and reusable code. If you have any questions or need further clarification, feel free to leave a comment below!

Read posts more at : Mastering Java Generics: A Comprehensive Guide with Code Examples

Top comments (0)