A股上市公司传智教育(股票代码 003032)旗下技术交流社区北京昌平校区

 找回密码
 加入黑马

QQ登录

只需一步,快速开始

© haixian 中级黑马   /  2014-4-3 19:51  /  1001 人查看  /  2 人回复  /   0 人收藏 转载请遵从CC协议 禁止商业使用本文

为什么泛型参数不能够是基本数据类型。

评分

参与人数 1技术分 +1 收起 理由
朱神必 + 1

查看全部评分

2 个回复

倒序浏览
泛型机制常用的参数有3个:
“?”代表任意类型。如果只指定了<?>,而没有extends,则默认是允许任意类。
extends关键字声明了类型的上界,表示参数化的类型可能是所指定的类型,或者是此类型的子类。
super关键字声明了类型的下界,表示参数化的类型可能是所指定的类型,或者是此类型的父类型,直至Object
前提
Fruit是Apple和Orange的超类。
本章通过java代码逐一分析泛型参数的意义和区别
extends参数:
[java]  
public void extend(List<? extends Fruit> list, Fruit fruit){   
        Fruit now = list.get(0);   
        System.out.println("now==>" + now);  
    //    Apple a = list.get(0);  无法通过编译  
        Object o = list.get(0);   
        System.out.println(o);  
        // list.add(fruit); //这句话无法编译      
        // list.add(new Object());  // 无法通过编译  
        list.add(null);//这句可以编译,因为null没有类型信息   
    }   
在上面的例子中,可以明确从list中取得的记录一定是Fruit,但是无法确定是Apple,list除了传入null外,不能添加其他任何参数;分析原因前先看以下的例子:
[java]
public void testExtendType(){  
    List<? extends Fruit> list = new ArrayList<Fruit>();  
    List<? extends Fruit> list2 = new ArrayList<Apple>();  
    List<? extends Fruit> list3 = new ArrayList<Orange>();  
    extend(list2, new Apple());  
}  
从以上代码可以看出:
extends关键字声明了类型的上界,表示参数化的类型可能是所指定的类型,或者是此类型的子类,List<? extends Fruit> 可以指向 ArrayList<Fruit>()、ArrayList<Apple>()、ArrayList<Orange>()等。现在如果传入的参数list为new ArrayList<Apple>(),则此时你向此list中添加
Fruit,则肯定报错。而读取数据时,不管list是什么类型,读取的数据肯定都是Fruit,同时Fruit也是Object。
通过以上的分析,知道直接往继承extends中的列表中添加记录是不行的,那么如何向extends修饰的list中添加记录,可以使用泛型方法来达到目的:
[java]  
public <T extends Fruit> void extendType2(List<T> list, T date){   
        list.add(date);   
    }   
调用示例如下:
[java]  
public void testExtendType2(){   
        List<Apple> list = new ArrayList<Apple>();   
        extendType2(list,new Apple());   
    }  
super参数
super用的不多,只简单介绍一下。
[java]  
public void superType(List<? super Apple> list, Apple apple){   
        Object o = list.get(0);  
        System.out.println(o);  
    //  Apple a = list.get(0); // 不能编译  
         
     // Apple apple = list.get(0); // 不能编译   
        list.add(apple);  
//      list.add( new Object());  // 不能编译  
//      list.add( new Fruit());  // 不能编译  
    }   
在以上代码中,从list中读取的记录无法确定是任何类型(除Object),list除了添加Apple外,不能添加任何类型的对象(包括Object)。分析原因前先看以下代码:
[java]  
public void testSuperType(){   
    List<? super Apple> list = new ArrayList<Apple>();   
    List<? super Apple> list2 = new ArrayList<Fruit>();   
    superType(list2, new Apple());   
}  
super关键字声明了类型的下界,表示参数化的类型是所指定的类型,或者是此类型的父类型,直至Object。List<? super Apple > 可以引用  List<Apple>,也可以引用 List<Fruit>,所以无法保证list中都是Apple,但是可以保证,所有list引用的都是Apple的父类,所有向此列表中添加Apple是没有问题的,而很显然你向List<Fruit>、List<Apple>中添加Object、Fruit对象,肯定失败,只能够添加Apple。
?参数
“?”代表任意类型。?可以看做 ? extends Object 的缩写。
[java]
public void wideCardType(List<?> list, Fruit fruit ){  
    Object o = list.get(0);  
    System.out.println(list.get(0));  
    //  list.add(fruit);  // 不能编译  
    //  list.add(new Object()); // 不能被编译  
}  
[java]  
public void testWideCardType(){  
        /**
         * ? 可以看做 ? extends Object 的简化版
         */  
        List<?> list = new ArrayList<Fruit>();  
        List<?> list2 = new ArrayList<Apple>();  
        List<?> list3 = new ArrayList<Object>();  
         
        wideCardType(list,new Fruit());  
    }  
List<? super Object> 可以引用  List<任何类型>,而Object是所有的对象的祖先,此list只能添加Object类型对象。
extends和super意义记忆的技巧
? extends A: ?继承A,?是A的子类,或者?是A本身
? super B:    ?是B的父类,或者?是B本身

评分

参与人数 1技术分 +2 收起 理由
朱神必 + 2

查看全部评分

回复 使用道具 举报
泛型是一种编译机制,在编译时期,任何泛型类型都会被转换成相应的原始类型,原始类型是由泛型类经过类型擦除得到的,类型擦除就是将类型变量替换为限定类型(无限定的变量用 Object),而基本数据类型没有限定类型且不能被替换为 Object,因此不能用于泛型。

评分

参与人数 1技术分 +1 收起 理由
朱神必 + 1

查看全部评分

回复 使用道具 举报
您需要登录后才可以回帖 登录 | 加入黑马