2015-09-07 11:44:36 -06:00
|
|
|
// generics/Wildcards.java
|
2016-12-30 17:23:13 -08:00
|
|
|
// (c)2017 MindView LLC: see Copyright.txt
|
2015-11-15 15:51:35 -08:00
|
|
|
// We make no guarantees that this code is fit for any purpose.
|
2016-09-23 13:23:35 -06:00
|
|
|
// Visit http://OnJava8.com for more book information.
|
2016-01-25 18:05:55 -08:00
|
|
|
// Exploring the meaning of wildcards
|
2015-06-15 17:47:35 -07:00
|
|
|
|
|
|
|
public class Wildcards {
|
|
|
|
// Raw argument:
|
|
|
|
static void rawArgs(Holder holder, Object arg) {
|
2016-01-25 18:05:55 -08:00
|
|
|
//- holder.set(arg);
|
|
|
|
// warning: [unchecked] unchecked call to set(T)
|
|
|
|
// as a member of the raw type Holder
|
|
|
|
// holder.set(arg);
|
|
|
|
// ^
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in class Holder
|
|
|
|
// 1 warning
|
2015-06-15 17:47:35 -07:00
|
|
|
|
|
|
|
// Can't do this; don't have any 'T':
|
|
|
|
// T t = holder.get();
|
|
|
|
|
|
|
|
// OK, but type information is lost:
|
|
|
|
Object obj = holder.get();
|
|
|
|
}
|
|
|
|
// Similar to rawArgs(), but errors instead of warnings:
|
|
|
|
static void unboundedArg(Holder<?> holder, Object arg) {
|
2016-01-25 18:05:55 -08:00
|
|
|
//- holder.set(arg);
|
|
|
|
// error: method set in class Holder<T>
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// holder.set(arg);
|
|
|
|
// ^
|
|
|
|
// required: CAP#1
|
|
|
|
// found: Object
|
|
|
|
// reason: argument mismatch;
|
|
|
|
// Object cannot be converted to CAP#1
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in class Holder
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Object from capture of ?
|
|
|
|
// 1 error
|
2015-06-15 17:47:35 -07:00
|
|
|
|
|
|
|
// Can't do this; don't have any 'T':
|
|
|
|
// T t = holder.get();
|
|
|
|
|
|
|
|
// OK, but type information is lost:
|
|
|
|
Object obj = holder.get();
|
|
|
|
}
|
|
|
|
static <T> T exact1(Holder<T> holder) {
|
|
|
|
return holder.get();
|
|
|
|
}
|
|
|
|
static <T> T exact2(Holder<T> holder, T arg) {
|
|
|
|
holder.set(arg);
|
|
|
|
return holder.get();
|
|
|
|
}
|
|
|
|
static <T>
|
|
|
|
T wildSubtype(Holder<? extends T> holder, T arg) {
|
2016-01-25 18:05:55 -08:00
|
|
|
//- holder.set(arg);
|
|
|
|
// error: method set in class Holder<T#2>
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// holder.set(arg);
|
|
|
|
// ^
|
|
|
|
// required: CAP#1
|
|
|
|
// found: T#1
|
|
|
|
// reason: argument mismatch;
|
|
|
|
// T#1 cannot be converted to CAP#1
|
|
|
|
// where T#1,T#2 are type-variables:
|
|
|
|
// T#1 extends Object declared in method
|
|
|
|
// <T#1>wildSubtype(Holder<? extends T#1>,T#1)
|
|
|
|
// T#2 extends Object declared in class Holder
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends T#1 from capture of ? extends T#1
|
|
|
|
// 1 error
|
|
|
|
|
2015-06-15 17:47:35 -07:00
|
|
|
return holder.get();
|
|
|
|
}
|
|
|
|
static <T>
|
|
|
|
void wildSupertype(Holder<? super T> holder, T arg) {
|
|
|
|
holder.set(arg);
|
2016-01-25 18:05:55 -08:00
|
|
|
//- T t = holder.get();
|
|
|
|
// error: incompatible types:
|
|
|
|
// CAP#1 cannot be converted to T
|
|
|
|
// T t = holder.get();
|
|
|
|
// ^
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in method
|
|
|
|
// <T>wildSupertype(Holder<? super T>,T)
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Object super:
|
|
|
|
// T from capture of ? super T
|
|
|
|
// 1 error
|
2015-06-15 17:47:35 -07:00
|
|
|
|
|
|
|
// OK, but type information is lost:
|
|
|
|
Object obj = holder.get();
|
|
|
|
}
|
|
|
|
public static void main(String[] args) {
|
|
|
|
Holder raw = new Holder<>();
|
|
|
|
// Or:
|
|
|
|
raw = new Holder();
|
|
|
|
Holder<Long> qualified = new Holder<>();
|
|
|
|
Holder<?> unbounded = new Holder<>();
|
|
|
|
Holder<? extends Long> bounded = new Holder<>();
|
|
|
|
Long lng = 1L;
|
|
|
|
|
|
|
|
rawArgs(raw, lng);
|
|
|
|
rawArgs(qualified, lng);
|
|
|
|
rawArgs(unbounded, lng);
|
|
|
|
rawArgs(bounded, lng);
|
|
|
|
|
|
|
|
unboundedArg(raw, lng);
|
|
|
|
unboundedArg(qualified, lng);
|
|
|
|
unboundedArg(unbounded, lng);
|
|
|
|
unboundedArg(bounded, lng);
|
|
|
|
|
2016-01-25 18:05:55 -08:00
|
|
|
//- Object r1 = exact1(raw);
|
|
|
|
// warning: [unchecked] unchecked method invocation:
|
|
|
|
// method exact1 in class Wildcards is applied
|
|
|
|
// to given types
|
|
|
|
// Object r1 = exact1(raw);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>
|
|
|
|
// found: Holder
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact1(Holder<T>)
|
|
|
|
// warning: [unchecked] unchecked conversion
|
|
|
|
// Object r1 = exact1(raw);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>
|
|
|
|
// found: Holder
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact1(Holder<T>)
|
|
|
|
// 2 warnings
|
|
|
|
|
2015-06-15 17:47:35 -07:00
|
|
|
Long r2 = exact1(qualified);
|
|
|
|
Object r3 = exact1(unbounded); // Must return Object
|
|
|
|
Long r4 = exact1(bounded);
|
|
|
|
|
2016-01-25 18:05:55 -08:00
|
|
|
//- Long r5 = exact2(raw, lng);
|
|
|
|
// warning: [unchecked] unchecked method invocation:
|
|
|
|
// method exact2 in class Wildcards is
|
|
|
|
// applied to given types
|
|
|
|
// Long r5 = exact2(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>,T
|
|
|
|
// found: Holder,Long
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact2(Holder<T>,T)
|
|
|
|
// warning: [unchecked] unchecked conversion
|
|
|
|
// Long r5 = exact2(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>
|
|
|
|
// found: Holder
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact2(Holder<T>,T)
|
|
|
|
// 2 warnings
|
|
|
|
|
2015-06-15 17:47:35 -07:00
|
|
|
Long r6 = exact2(qualified, lng);
|
2016-01-25 18:05:55 -08:00
|
|
|
|
|
|
|
//- Long r7 = exact2(unbounded, lng);
|
|
|
|
// error: method exact2 in class Wildcards
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// Long r7 = exact2(unbounded, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>,T
|
|
|
|
// found: Holder<CAP#1>,Long
|
|
|
|
// reason: inference variable T has
|
|
|
|
// incompatible bounds
|
|
|
|
// equality constraints: CAP#1
|
|
|
|
// lower bounds: Long
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact2(Holder<T>,T)
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Object from capture of ?
|
|
|
|
// 1 error
|
|
|
|
|
|
|
|
//- Long r8 = exact2(bounded, lng);
|
|
|
|
// error: method exact2 in class Wildcards
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// Long r8 = exact2(bounded, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<T>,T
|
|
|
|
// found: Holder<CAP#1>,Long
|
|
|
|
// reason: inference variable T
|
|
|
|
// has incompatible bounds
|
|
|
|
// equality constraints: CAP#1
|
|
|
|
// lower bounds: Long
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>exact2(Holder<T>,T)
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Long from
|
|
|
|
// capture of ? extends Long
|
|
|
|
// 1 error
|
|
|
|
|
|
|
|
//- Long r9 = wildSubtype(raw, lng);
|
|
|
|
// warning: [unchecked] unchecked method invocation:
|
|
|
|
// method wildSubtype in class Wildcards
|
|
|
|
// is applied to given types
|
|
|
|
// Long r9 = wildSubtype(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? extends T>,T
|
|
|
|
// found: Holder,Long
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSubtype(Holder<? extends T>,T)
|
|
|
|
// warning: [unchecked] unchecked conversion
|
|
|
|
// Long r9 = wildSubtype(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? extends T>
|
|
|
|
// found: Holder
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSubtype(Holder<? extends T>,T)
|
|
|
|
// 2 warnings
|
|
|
|
|
2015-06-15 17:47:35 -07:00
|
|
|
Long r10 = wildSubtype(qualified, lng);
|
|
|
|
// OK, but can only return Object:
|
|
|
|
Object r11 = wildSubtype(unbounded, lng);
|
|
|
|
Long r12 = wildSubtype(bounded, lng);
|
|
|
|
|
2016-01-25 18:05:55 -08:00
|
|
|
//- wildSupertype(raw, lng);
|
|
|
|
// warning: [unchecked] unchecked method invocation:
|
|
|
|
// method wildSupertype in class Wildcards
|
|
|
|
// is applied to given types
|
|
|
|
// wildSupertype(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? super T>,T
|
|
|
|
// found: Holder,Long
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSupertype(Holder<? super T>,T)
|
|
|
|
// warning: [unchecked] unchecked conversion
|
|
|
|
// wildSupertype(raw, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? super T>
|
|
|
|
// found: Holder
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSupertype(Holder<? super T>,T)
|
|
|
|
// 2 warnings
|
|
|
|
|
2015-06-15 17:47:35 -07:00
|
|
|
wildSupertype(qualified, lng);
|
2016-01-25 18:05:55 -08:00
|
|
|
|
|
|
|
//- wildSupertype(unbounded, lng);
|
|
|
|
// error: method wildSupertype in class Wildcards
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// wildSupertype(unbounded, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? super T>,T
|
|
|
|
// found: Holder<CAP#1>,Long
|
|
|
|
// reason: cannot infer type-variable(s) T
|
|
|
|
// (argument mismatch; Holder<CAP#1>
|
|
|
|
// cannot be converted to Holder<? super T>)
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSupertype(Holder<? super T>,T)
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Object from capture of ?
|
|
|
|
// 1 error
|
|
|
|
|
|
|
|
//- wildSupertype(bounded, lng);
|
|
|
|
// error: method wildSupertype in class Wildcards
|
|
|
|
// cannot be applied to given types;
|
|
|
|
// wildSupertype(bounded, lng);
|
|
|
|
// ^
|
|
|
|
// required: Holder<? super T>,T
|
|
|
|
// found: Holder<CAP#1>,Long
|
|
|
|
// reason: cannot infer type-variable(s) T
|
|
|
|
// (argument mismatch; Holder<CAP#1>
|
|
|
|
// cannot be converted to Holder<? super T>)
|
|
|
|
// where T is a type-variable:
|
|
|
|
// T extends Object declared in
|
|
|
|
// method <T>wildSupertype(Holder<? super T>,T)
|
|
|
|
// where CAP#1 is a fresh type-variable:
|
|
|
|
// CAP#1 extends Long from capture of
|
|
|
|
// ? extends Long
|
|
|
|
// 1 error
|
2015-06-15 17:47:35 -07:00
|
|
|
}
|
2015-09-07 11:44:36 -06:00
|
|
|
}
|