Generics deal with type-safe objects making the code stable by detecting the bug at compile time.
Before Generics was introduced, we can store any type of data in the collection. Now, in Generics, we can store only a specific type of object.
Advantages of Generics:
1] Type-Safety: We can now store only a specific type of object. We cannot store other objects in the collection.
2] We do not need to typecast the object, so Typecasting is not required.
3] It can detect the errors at compile time itself, and hence the problem does not occur at run time.
4] Type Erasure: Any extra information added using Generics into source code will get removed as soon as its byte code is generated.
Type Parameters:
The standard type parameters are:
T ā Type
K ā Key
E ā Element
V ā Value
N ā Number
Generic Class:
When more than one type of parameter is declared in a class, it is said to be as Generic Class.
class DemoClassĀ { Ā Ā Ā private Object t; Ā Ā Ā public void set(Object t)Ā {Ā this.t = t;Ā } Ā Ā Ā public Object get()Ā { Ā return t;Ā } }
Generic Method:
Generic Method is the same as Generic Class, where more than one type of parameter can be declared. The only difference between the Generic method and Generic class is that the scope of a type parameter resides within its method.
public static <T> int countAllOccurrences(T[] list, T item) { Ā Ā Ā int count = 0; Ā Ā Ā if (item == null) { Ā Ā Ā Ā Ā Ā for ( T listItem : list ) Ā Ā Ā Ā Ā Ā Ā Ā Ā if (listItem == null) Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā count++; Ā Ā Ā } Ā Ā Ā else { Ā Ā Ā Ā Ā Ā for ( T listItem : list ) Ā Ā Ā Ā Ā Ā Ā Ā Ā if (item.equals(listItem)) Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā count++; Ā Ā Ā } Ā Ā Ā return count; } Ā
Generics With Wildcards:
The ? (question mark) denotes the wildcard. It can be used as a type of a parameter, a local variable, field or return type. Examples of wildcard parameterized types includes Collection<?<, List<? extends Number<, Comparator<? super String> and Pair<String,?>. Wildcards can be either bounded or unbounded.
abstract class Shaper{Ā Ā abstract void draw();Ā Ā }Ā Ā class Triangle extends Shaper{Ā Ā void draw(){System.out.println("drawing Triangle");}Ā Ā }Ā Ā class Square extends Shaper{Ā Ā void draw(){System.out.println("drawing Square");}Ā Ā }Ā Ā class GenericTest{Ā Ā public static void drawShapers(List<? extends Shaper> lists){Ā Ā for(Shaper s:lists){Ā Ā s.draw();//calling method of Shaper class by child class instanceĀ Ā }Ā Ā }Ā Ā public static void main(String args[]){Ā Ā List<Triangle> list1=new ArrayList<Triangle>();Ā Ā list1.add(new Triangle());Ā Ā Ā Ā List<Square> list2=new ArrayList<Square>();Ā Ā list2.add(new Square());Ā Ā list2.add(new Square());Ā Ā Ā Ā drawShapers(list1);Ā Ā drawShapers(list2);Ā Ā }}Ā
Bounded Wildcards:
Bounded wildcards have some restrictions on type parameters. This restriction can be enforced using super and extends keyword.
1] Upper Bounded Wildcards: This is used to decrease the restriction on a variable. It is declared by using wildcard character following by extends or implements and then following by its upper bound.
public class UpperBoundWildcard {Ā Ā Ā Ā Ā Ā private static Double add(ArrayList<? extends Number> num) {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā double sum=0.0;Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā for(Number n:num)Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā sum = sum+n.doubleValue();Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā }Ā Ā Ā Ā Ā Ā Ā Ā Ā return sum;Ā Ā Ā Ā Ā Ā }Ā Ā Ā Ā Ā Ā public static void main(String[] args) {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā ArrayList<Integer> l1=new ArrayList<Integer>();Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā l1.add(10);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā l1.add(20);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā System.out.println("displaying the sum= "+add(l1));Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā ArrayList<Double> l2=new ArrayList<Double>();Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā l2.add(30.0);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā l2.add(40.0);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā System.out.println("displaying the sum= "+add(l2));Ā Ā Ā Ā Ā Ā Ā Ā Ā }Ā }
2] Lower Bounded Wildcards: This is used to decrease the restriction on the unknown type to be a specific type or a supertype. It is declared by using wildcard character following by super keyword and then following by it lower bound.
public class LowerBoundWildcard {Ā Ā Ā Ā Ā Ā public static void addNumbers(List<? super Integer> list) {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā for(Object n:list)Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā System.out.println(n);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā }Ā Ā Ā Ā Ā Ā }Ā Ā public static void main(String[] args) {Ā Ā Ā Ā Ā Ā List<Integer> l1=Arrays.asList(1,2,3);Ā Ā Ā Ā Ā Ā Ā Ā System.out.println("displaying the Integer values");Ā Ā Ā Ā Ā Ā addNumbers(l1);Ā Ā Ā Ā Ā Ā List<Number> l2=Arrays.asList(1.0,2.0,3.0);Ā Ā Ā Ā Ā Ā Ā Ā System.out.println("displaying the Number values");Ā Ā Ā Ā Ā Ā addNumbers(l2);Ā Ā }Ā Ā }Ā
Unbounded Wildcards:
It represents the list of an unknown type say for eg. List<?>. Unbounded wildcards are useful in two cases:
- When the method is implemented in the Object Class.
- When Generic Class contains those methods which do not depend on the type parameter.
public class UnboundedWildcard {Ā Ā Ā Ā Ā Ā public static void display(List<?> list)Ā Ā Ā Ā Ā Ā {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā for(Object o:list)Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā {Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā System.out.println(o);Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā }Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā }Ā Ā Ā Ā Ā Ā public static void main(String[] args) {Ā Ā Ā Ā Ā Ā List<Integer> l1=Arrays.asList(1,2,3);Ā Ā Ā Ā Ā Ā System.out.println("displaying the Integer values");Ā Ā Ā Ā Ā Ā display(l1);Ā Ā Ā Ā Ā Ā List<String> l2=Arrays.asList("One","Two","Three");Ā Ā Ā Ā Ā Ā Ā Ā System.out.println("displaying the String values");Ā Ā Ā Ā Ā Ā Ā Ā Ā Ā display(l2);Ā Ā Ā Ā Ā Ā }Ā Ā }Ā