[Java]泛型的extends和super
代碼
import java.util.*;
public class T1 {
static class View {}
static class ViewGroup extends View {}
static class LinearLayout extends ViewGroup {}
static class MyLayout extends LinearLayout {}
static class FrameLayout extends ViewGroup {}
public static void main(String[] args) {
List<? extends ViewGroup> view1 = new ArrayList<FrameLayout>();// 可以傳遞的泛型是 ViewGroup LinearLayout MyLayout FrameLayout
view1.add(null);
view1.add(new Object());//compile error
view1.add(new View());//compile error
view1.add(new ViewGroup());//compile error
view1.add(new LinearLayout());//compile error
view1.add(new MyLayout());//compile error
view1.add(new FrameLayout());//compile error
ViewGroup v = view1.get(0);
//下界
List<? super ViewGroup> view2 = new ArrayList<ViewGroup>();//可以傳遞的泛型是 Object View ViewGroup
view2.add(null);
view2.add(new Object());//compile error
view2.add(new View());//compile error
view2.add(new ViewGroup());
view2.add(new LinearLayout());
view2.add(new MyLayout());
view2.add(new FrameLayout());
Object object = view2.get(0);
}
public static void test1(List<? extends ViewGroup> v) {
ViewGroup vg = v.get(0);
}
public static void test2(List<ViewGroup> v) {
ViewGroup vg = v.get(0);
}
public static void test3(ViewGroup vg) {
}
}
view1 的泛型爲<? extends ViewGroup>
,那麼它可以接受的值可以是
new XXXList<ViewGroup>();
new XXXList<LinearLayout>();//LinearLayout 繼承自ViewGroup
new XXXList<MyLayout>();//MyLayout 繼承自LinearLayout
new XXXList<FrameLayout>();//FrameLayout 繼承自VewGroup
但是view1 調用add
就會全部出錯。
比如view1.add(new ViewGroup());
,雖然泛型爲<? extends ViewGroup>
,但是view1的值是不能夠確定,因爲它還可以是new XXXList<LinearLayout>();
,父類不能夠轉換成子類,只能子類轉換成父類,同樣的,傳遞LinearLayout 也不行,因爲view1 的值也可能是new XXXList<MyLayout>();
,其他的同理。至於Object
和View
本來就不太行。
view2的泛型爲<? super ViewGroup>
,那麼它可以接受的值可以是
new XXXList<Object>();
new XXXList<View>();
new XXXList<ViewGroup>();
現在我們調用view2.add(new Object());
,雖然view2可能是new XXXList<Object>();
,但它也可能是new XXXList<View>();
,所以不成功。同理,調用view2.add(new View());
,也不行。
但是view2.add(new ViewGroup());
和後面的都可以。因爲,不管view2的值是上面3種中的哪一種值,都可以進行安全的轉換即子類轉換成父類。
比如傳遞LinearLayout
,view2=new XXXList<Object>();
沒有問題,因爲LinearLayout
可以轉換成Object
,view2=new XXXList<View>();
也沒有問題,因爲LinearLayout
也可以轉換成View
。假如view2=new XXXList<ViewGroup>();
也沒有問題,因爲LinearLayout
也可以轉換成ViewGroup
。
但是傳遞View
就不行了,view2
也可能是new XXXList<ViewGroup>();
。雖然在view2=new XXXList<View>();
時可行,但是view2也可能時new XXXList<ViewGroup>();
。所以這個可能會出現問題,不可行。
使用
查看上面的代碼,view1
幾乎是“不可寫”,不管你傳遞哪個都不行,它到底有什麼用?它的用處主要在參數傳遞,比如上面的test1(List<? extends ViewGroup> fruits)
,我可以傳遞的值爲
new XXXList<ViewGroup>();
new XXXList<LinearLayout>();
new XXXList<MyLayout>();
new XXXList<FrameLayout>();
但是,test2(List<ViewGroup> v)
就只能夠傳遞new XXXList<ViewGroup>()
。