Thursday, October 31, 2013

More Fun with Wildcards

In this section, we'll consider some of the more advanced uses of wildcards. We've seen several examples where bounded wildcards were useful when reading from a data structure. Now consider the inverse, a write-only data structure. The interface Sink is a simple example of this sort.
interface Sink {
    flush(T t);
}
We can imagine using it as demonstrated by the code below. The method writeAll() is designed to flush all elements of the collection coll to the sink snk, and return the last element flushed.
public static  T writeAll(Collection coll, Sink snk) {
    T last;
    for (T t : coll) {
        last = t;
        snk.flush(last);
    }
    return last;
}
...
Sink s;
Collection cs;
String str = writeAll(cs, s); // Illegal call.
As written, the call to writeAll() is illegal, as no valid type argument can be inferred; neither String nor Object are appropriate types for T, because the Collectionelement and the Sink element must be of the same type.
We can fix this error by modifying the signature of writeAll() as shown below, using a wildcard.
public static  T writeAll(Collectionextends
T>, Sink) {...} ... // Call is OK, but wrong return type. String str = writeAll(cs, s); The call is now legal, but the assignment is erroneous, since the return type inferred is Object because T matches the element type of s, which is Object.
The solution is to use a form of bounded wildcard we haven't seen yet: wildcards with a lower bound. The syntax super T denotes an unknown type that is a supertype of T(or T itself; remember that the supertype relation is reflexive). It is the dual of the bounded wildcards we've been using, where we use extends T to denote an unknown type that is a subtype of T.
public static  T writeAll(Collection coll, Sinksuper
T> snk) { ... } String str = writeAll(cs, s); // Yes! Using this syntax, the call is legal, and the inferred type is String, as desired.
Now let's turn to a more realistic example. A java.util.TreeSet represents a tree of elements of type E that are ordered. One way to construct a TreeSet is to pass aComparator object to the constructor. That comparator will be used to sort the elements of the TreeSet according to a desired ordering.
TreeSet(Comparator c) 
The Comparator interface is essentially:
interface Comparator {
    int compare(T fst, T snd);
}
Suppose we want to create a TreeSet and pass in a suitable comparator, We need to pass it a Comparator that can compare Strings. This can be done by aComparator, but a Comparator will do just as well. However, we won't be able to invoke the constructor given above on a Comparator. We can use a lower bounded wildcard to get the flexibility we want:
TreeSet(Comparatorsuper
E> c) This code allows any applicable comparator to be used.
As a final example of using lower bounded wildcards, lets look at the method Collections.max(), which returns the maximal element in a collection passed to it as an argument. Now, in order for max() to work, all elements of the collection being passed in must implement Comparable. Furthermore, they must all be comparable to each other.
A first attempt at generifying this method signature yields:
public static extends
Comparable> T max(Collection coll) That is, the method takes a collection of some type T that is comparable to itself, and returns an element of that type. However, this code turns out to be too restrictive. To see why, consider a type that is comparable to arbitrary objects:
class Foo implements Comparable {
    ...
}
Collection cf = ... ;
Collections.max(cf); // Should work.
Every element of cf is comparable to every other element in cf, since every such element is a Foo, which is comparable to any object, and in particular to another Foo. However, using the signature above, we find that the call is rejected. The inferred type must be Foo, but Foo does not implement Comparable.
It isn't necessary that T be comparable to exactly itself. All that's required is that T be comparable to one of its supertypes. This give us:
public static extends
Comparablesuper T>> T max(Collection coll) Note that the actual signature of Collections.max() is more involved. We return to it in the next section, Converting Legacy Code to Use Generics. This reasoning applies to almost any usage of Comparable that is intended to work for arbitrary types: You always want to use Comparablesuper
 T>.In general, if you have an API that only uses a type parameter T as an argument, its uses should take advantage of lower bounded wildcards (super T). Conversely, if the API only returns T, you'll give your clients more flexibility by using upper bounded wildcards (extends T).

Wildcard Capture

It should be pretty clear by now that given:
Set unknownSet = new HashSet();
...
/* Add an element  t to a Set s. */ 
public static  void addToSet(Set s, T t) {
    ...
}
The call below is illegal.
addToSet(unknownSet, "abc"); // Illegal.
It makes no difference that the actual set being passed is a set of strings; what matters is that the expression being passed as an argument is a set of an unknown type, which cannot be guaranteed to be a set of strings, or of any type in particular.
Now, consider the following code:
class Collections {
    ...
     public static Set unmodifiableSet(Set set) {
        ...
    }
}
...
Set s = Collections.unmodifiableSet(unknownSet); // This works! Why?
It seems this should not be allowed; yet, looking at this specific call, it is perfectly safe to permit it. After all, unmodifiableSet() does work for any kind of Set, regardless of its element type.
Because this situation arises relatively frequently, there is a special rule that allows such code under very specific circumstances in which the code can be proven to be safe. This rule, known as wildcard capture, allows the compiler to infer the unknown type of a wildcard as a type argument to a generic method.

No comments: