在泛型应用中得特别小心,否则会出现一些莫名奇妙的错误信息,例如
public interface Box<T>{
public T get();
public void put(T content);
}
在如下使用该接口时
public void dobox(Box<?> box){
box.put(box.get());
}
该方法看上去可以运行,但实际上不行,编译器会提示(大概意思):"capture#453 of ? not being compatible with Object "
其中capture#453 of ? 是一个占位符,是编译器为该例通配符"?"指定的一个名字.
要使用好泛型,可以注意下面两个诀窍:
1、capture helper
针对dobox()方法可以这样修改一下:
public void dobox(Box<?> box){
doboxHelper(box);
}
private<V> void doboxHelper(Box<V> box){
box.put(box.get());
}
这样这段代码可以运行了,因为此时在doboxHelper()中,box.get()返回的类型不是Object,而是类型V
2、generic factory
一般在构造泛型实例时这样操作
Box<String> box= new Box<String>;
这违反了DRY原则(Don't Repeat Yourself ),从而可以这样修改一下:
public class BoxFactory<T> implements Box<T>{
public static<T> Box<T> make(){
return new BoxFactory<T>();
}
......
}
在这里用到了编译器只能在泛型方法中进行推断类型参数的特性!先写到这里了!
public interface Box<T>{
public T get();
public void put(T content);
}
在如下使用该接口时
public void dobox(Box<?> box){
box.put(box.get());
}
该方法看上去可以运行,但实际上不行,编译器会提示(大概意思):"capture#453 of ? not being compatible with Object "
其中capture#453 of ? 是一个占位符,是编译器为该例通配符"?"指定的一个名字.
要使用好泛型,可以注意下面两个诀窍:
1、capture helper
针对dobox()方法可以这样修改一下:
public void dobox(Box<?> box){
doboxHelper(box);
}
private<V> void doboxHelper(Box<V> box){
box.put(box.get());
}
这样这段代码可以运行了,因为此时在doboxHelper()中,box.get()返回的类型不是Object,而是类型V
2、generic factory
一般在构造泛型实例时这样操作
Box<String> box= new Box<String>;
这违反了DRY原则(Don't Repeat Yourself ),从而可以这样修改一下:
public class BoxFactory<T> implements Box<T>{
public static<T> Box<T> make(){
return new BoxFactory<T>();
}
......
}
在这里用到了编译器只能在泛型方法中进行推断类型参数的特性!先写到这里了!