12

I understand in

Comparator < ? super T> comp

it returns the maximum element of the given collection, according to the order defined by the specified comparator. But I don't understand the purpose of the

super T

Could anyone possibly explain?

Matthias Braun
  • 28,341
  • 18
  • 134
  • 157
Marcello
  • 413
  • 1
  • 5
  • 12

3 Answers3

20

The term ? super T means "unknown type that is, or is a super class of, T", which in generics parlance means its lower bound is T.

This signature is used because T may be assigned to, and compared with, any variable whose type is, or is a super class of, T. Ie if a Comparator can accept a super class of T in its compare() method, you can pass in a T.

This follows the PECS mnemonic: "Producer Extends, Consumer Super", which means that producers of things should work with things that have an upper bound ( ? extends T) and consumers (like comparator implementations that use things) should work with lower bounds ( ? super T).

Matthias Braun
  • 28,341
  • 18
  • 134
  • 157
Bohemian
  • 389,931
  • 88
  • 552
  • 692
6

In here < ? super T> means generics - NOT comparisons.

It means you have a Comparator with a generic type of ? super T (something that extends is super typed by T), as explained in this thread

comp is the variable name (binding).

So basically in here Comparator < ? super T> is a type and comp is an identifier (variable name), that is of type Comparator <? super T>


For more info: Java Generics

Matthias Braun
  • 28,341
  • 18
  • 134
  • 157
amit
  • 172,148
  • 26
  • 225
  • 324
-1

Let's use the type of one of TreeSet's constructors as an example:

TreeSet(Comparator<? super E> comparator)

Imagine that E is a String, so we mentally specialize the constructor:

TreeSet(Comparator<? super String> comparator)

I can instantiate TreeSet like this:

var mySet = new TreeSet<String>(myStringComparator);

Here, myStringComparator might look like this.

Comparator<String> myStringComparator = new Comparator<>() {
    @Override
    public int compare(String o1, String o2) {
        return 0;
    }
};

The <? super String> part of TreeSet(Comparator<? super String> comparator) allows us to also provide the constructor with a comparator that compares super types of Strings. That's a more general, less-specific, comparator you might say.

For example, CharSequence is a super type of String, that's why this works, too:

Comparator<CharSequence> myCharSeqComparator = new Comparator<>() {
    @Override
    public int compare(CharSequence o1, CharSequence o2) {
        return 0;
    }
};

var mySet2 = new TreeSet<String>(myCharSeqComparator);

Hypothetically, if TreeSet's constructor were defined like this:

TreeSet(Comparator<E> comparator)

which in our example specializes to

TreeSet(Comparator<String> comparator)

we could only provide Comparator<String>, but not Comparator<CharSequence> or Comparator<Object>.

This would be a pity since we might want to use the more general comparators that don't need the specifics of String to be able to compare.

Matthias Braun
  • 28,341
  • 18
  • 134
  • 157