Bounded Types and Wildcards (? extends T, ? super T)


Bounded types and wildcards in Java generics are powerful concepts that enable more flexible and reusable code. They provide fine-grained control over type parameters in generic classes, methods, and interfaces. In this article, we will explore the concepts of bounded types and wildcards (? extends T, ? super T) and provide examples to demonstrate their usage in advanced Java.

Step-by-Step Guide

Step 1: Understanding Bounded Type Parameters

In Java, you can restrict the types that can be used as type parameters. This restriction is called a bounded type parameter. Bounded types are specified using the extends keyword for upper bounds, meaning the type parameter must be a subclass of a given type.

Example: Bounded type parameter with an upper bound

            // Bounded type parameter example
            class Box<T extends Number> {
                private T value;

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

                public T getValue() {
                    return value;
                }
            }

            public class BoundedTypeExample {
                public static void main(String[] args) {
                    Box<Integer> integerBox = new Box<>();
                    integerBox.setValue(10);
                    System.out.println("Integer Value: " + integerBox.getValue());

                    Box<Double> doubleBox = new Box<>();
                    doubleBox.setValue(10.5);
                    System.out.println("Double Value: " + doubleBox.getValue());

                    // The following will cause a compile-time error
                    // Box<String> stringBox = new Box<>(); // Error: String is not a subclass of Number
                }
            }
        

Step 2: Using Wildcards with Generics

Wildcards in Java generics are used to represent an unknown type. The most common wildcard types are ? extends T (upper bounded wildcard) and ? super T (lower bounded wildcard). These wildcards allow you to work with generics without knowing the exact type.

Step 3: Using ? extends T (Upper Bounded Wildcard)

The ? extends T wildcard is used when you want to allow a method or class to accept types that are either of type T or a subtype of T. This is useful when you want to read data from a structure without modifying it.

Example: Using ? extends T to read elements of a list

            import java.util.*;

            public class UpperBoundedWildcardExample {

                // Method that accepts a list of elements that are instances of Number or its subclasses
                public static void printNumbers(List<? extends Number> list) {
                    for (Number num : list) {
                        System.out.println(num);
                    }
                }

                public static void main(String[] args) {
                    List<Integer> intList = Arrays.asList(1, 2, 3, 4);
                    List<Double> doubleList = Arrays.asList(1.1, 2.2, 3.3);

                    printNumbers(intList);   // Valid: Integer is a subclass of Number
                    printNumbers(doubleList); // Valid: Double is a subclass of Number
                }
            }
        

Step 4: Using ? super T (Lower Bounded Wildcard)

The ? super T wildcard is used when you want to allow a method or class to accept types that are either of type T or a supertype of T. This is useful when you want to add elements to a structure but don't know the exact type, as it guarantees that the structure can accept T or any superclass.

Example: Using ? super T to add elements to a list

            import java.util.*;

            public class LowerBoundedWildcardExample {

                // Method that accepts a list of elements that are instances of Number or its superclasses
                public static void addNumbers(List<? super Integer> list) {
                    list.add(10);  // Valid: Integer is a subclass of Number
                    list.add(20);  // Valid: Integer is a subclass of Number
                }

                public static void main(String[] args) {
                    List<Number> numberList = new ArrayList<>();
                    addNumbers(numberList);  // Valid: Number is a superclass of Integer

                    List<Object> objectList = new ArrayList<>();
                    addNumbers(objectList);  // Valid: Object is a superclass of Integer

                    // The following will cause a compile-time error
                    // List<Double> doubleList = new ArrayList<>();
                    // addNumbers(doubleList); // Error: Double is not a superclass of Integer
                }
            }
        

Step 5: Combining Bounded Wildcards

You can also combine ? extends T and ? super T in more complex scenarios. For example, you might have a method that reads from one list (using ? extends T) and adds to another (using ? super T).

Example: Combining ? extends T and ? super T

            import java.util.*;

            public class CombinedWildcardExample {

                // Method to copy elements from one list to another
                public static void copyList(List<? extends Number> source, List<? super Number> destination) {
                    for (Number num : source) {
                        destination.add(num); // Add elements to the destination list
                    }
                }

                public static void main(String[] args) {
                    List<Integer> intList = Arrays.asList(1, 2, 3);
                    List<Number> numberList = new ArrayList<>();

                    copyList(intList, numberList); // Valid: Integer is a subclass of Number

                    for (Number num : numberList) {
                        System.out.println(num); // Output: 1, 2, 3
                    }
                }
            }
        

Step 6: Best Practices for Using Wildcards and Bounded Types

  • Use ? extends T when you need to read from a structure, but don't want to modify it.
  • Use ? super T when you need to add elements to a structure, but don't want to restrict the supertype.
  • Always ensure the wildcard type is compatible with the expected operations (reading or writing) on the data.
  • Be cautious when combining multiple wildcard types to ensure the code remains readable and maintainable.

Conclusion

Bounded types and wildcards (? extends T, ? super T) in Java generics provide a way to write flexible, reusable, and type-safe code. By understanding and using these concepts, you can write methods and classes that work with a wide variety of types while ensuring type safety. Wildcards and bounded types are crucial for writing generic code that is both efficient and maintainable.





Advertisement