开发者

Compiler errors with java bounded wildcard generics

Having the below code:

    Stack<Integer> integers = new Stack<Integer>();
    Stack<? extends Number> numbers = integers;
    Number n = numbers.pop();
    numbers.push(3);
    numbers.push(n);

I get compilation errors on the last two lines, but although I've given it some thought I don't understand why the compilation errors are there.

The method开发者_如何学Python push(capture#2-of ? extends Number) in the type Stack<capture#2-of ? extends Number> is not applicable for the arguments (int)

When I comment the last line, I still get the above compilation error, but from my understanding the compiler should be able to infer the correct type (Stack) from those lines.

Thanks a lot


The last two lines aren't valid because numbers could be a stack of any numeric type. Consider this similar code:

Stack<Double> doubles = new Stack<Double>();
Stack<? extends Number> numbers = doubles;
Number n = numbers.pop();
numbers.push(3);
numbers.push(n);

Here you're trying to put an Integer onto a Stack<Double> which obviously isn't correct.

Basically when you use wildcarding like this, you can get values out, but you can't put values in because the compiler can't guarantee it's valid to do so.


Generics covariance in Java is handled at the client. i.e. You don't have the semantics to say that a Stack is covariant and let the compiler check that the operations you are allowing (like push) is valid in a covariance model. (push is not).

The specific issue here is that you can do something like this:

Number r = new Rational(a,b); // rationals are also numbers
number.push(r);

which in the underlying structure implies integer.push(r); // type mismatch

(Programming in Scala has a crystal clear explanation of client vs provider side (co-/contra-)variance on Chapter 19. A recommended read even if you are not into Scala)


very clear answer here How can I add to List<? extends Number> data structures?

0

上一篇:

下一篇:

精彩评论

暂无评论...
验证码 换一张
取 消

最新问答

问答排行榜