java基础——泛型

Java泛型(generics)是JDK 5中引入的一个新特性,允许在定义类/接口/方法的时候使用类型参数(type parameter)。声明的类型参数在使用时用具体的类型来替换。

1. 为什么使用泛型

泛型的好处在于编写重用性更好的代码,用来指定容器要持有什么类型的对象,由编译器保证类型的正确性,不需要强制类型转换。

2. 基本使用

2.1 泛型术语

  • List<E>被称作泛型类型
  • List<E>中的E被称为类型变量或类型参数
  • List<String>被称为参数化类型。
  • List<String>中的String被称为实际类型参数。
  • List被称为原始类型。

2.2 泛型类

创建类时,指明想持有的对象,将其置于尖括号之内。常见使用情景元组类库:是一个单一对象,将一组对象直接打包存储于其中。这个容器对象允许读取其中的元素,但是不允许向其中存放新的对象。

public class TwoTuple<A, B> {
    public final A first;
    public final B second;
    public TwoTuple(A a, B b) {  
        first = a;
        second = b;
    }
}

泛型接口定义类似于泛型类在这里不具体描述。

2.3 泛型方法

泛型方法使得该方法能够独立于类而产生变化,建议在使用泛型方法可以取代整个类泛型话的时候选择泛型方法。定义泛型方法,只需要泛型参数列表置于返回值之前。

public class Generators{
    public static <T> Collection<T> fill(Collection<T> coll, Generator<T> gne, int n){
        for(int i = 0; i < n;i++){
            coll.add(gen.next());
        }
        return coll;
    }
}

泛型方法使用注意点:

  1. 定义方法所用的泛型参数需要在修饰符之后添加,如 public static <T>,如果有多个泛型参数,可如此定义<K,V>。
  2. 类型参数推断,区别于泛型类在使用时候必须指定类型参数值,泛型方法则不必,编译器可以为我们找出具体类型,可以像调用普通方法一样调用fill()方法。

3. 泛型擦除

java的泛型是在编译器层面实现的,生成的字节码中是不包含泛型中的类型信息的

在JAVA的虚拟机中并不存在泛型,泛型只是为了完善java体系,增加程序员编程的便捷性以及安全性而创建的一种机制,在JAVA虚拟机中对应泛型的都是确定的类型。
在编写泛型代码后,java虚拟中会把这些泛型参数类型都擦除,用相应的确定类型来代替,代替的这一动作叫做类型擦除,而用于替代的类型称为原始类型,在类型擦除过程中,一般使用第一个限定的类型来替换,若无限定则使用Object。

有界类擦除前:

public class Eraser {    
    public static <A extends Comparable<A>> A max(Collection<A> xs) {        
        Iterator<A> xi = xs.iterator();        
        A w = xi.next();        
        while (xi.hasNext()) {            
            A x = xi.next();            
            if (w.compareTo(x) < 0)                
                w = x;        
            }        
        return w;    
    }
}

有界类擦除后:

public class Eraser {    
    public static Comparable max(Collection xs) {        
        Iterator xi = xs.iterator();        
        Comparable w = (Comparable)xi.next();        
        while (xi.hasNext()) {            
            Comparable x = (Comparable)xi.next();            
            if (w.compareTo(x) < 0)                
                w = x;        
            }        
        return w;    
    }
}

类型参数的边界<A extends Comparable<A>>A,A必须为Comparable<A>的子类,按照类型擦除的过程,先将所有的类型参数转换为最左边界Comparable<A>,然后去掉参数类型A,得到最终的擦除后结果。

多边界擦除
规则:使用其最左限制的擦除,即用第一个边界的类型变量来替换。
如果类Pair声明如下:
public class Pair<T extends Comparable & Serializable> { }
那么原始类型就是Comparable

4. 边界与通配符

边界使得你可以用于泛型的参数类型上设置限制条件,潜在效果你可以按照自己边界类型调用方法。

4.1 边界定义

<T extends BoundingType>这表示T应该是BoundingType的子类型,T和BoundingType可以是类,也可以是接口。另外,此处的extends表示的是子类型,不等同于继承。多个边界格式为:<T extends A & B & C>

示例代码中T可以在编译器调用HasColor拥有的方法。

class Colored<T extends HasColor> {    
    T item;    
    Colored(T item) {        
        this.item = item;    
    }    
    T getItem() {        
        return item;    
    }    
    //定义边界之后,如下的方法调用是合法的    
    java.awt.Color color() {        
        return item.getColor();    
    }
}
4.2 边界限定通配符
  • <? extends T> 表示类型的上界,表示参数化类型的可能是T或是T的子类。
  • <? super T> 表示类型下界,表示参数化类型是此类型的超类型(父类型),直至Object
  • <?> 表示类型无界,常见使用多个泛型参数允许其中某些为任何类型,例如:Map<String,?>

使用原则参考《Effective Java》:

  1. 如果要从集合中读取类型T的数据,并且不能写入,可以使用 ? extends 通配符;(Producer Extends)
  2. 如果要从集合中写入类型T的数据,并且不需要读取,可以使用 ? super 通配符;(Consumer Super)
  3. 如果既要存又要取,那么就不要使用无界通配符。
// demo1使用上界写入会编译错误
List<? extends Fruit> frutis = new ArrayList<Apple>();
fruits.add(new Apple());// complie error
fruits.add(new Fruit());// complie error
// demo2使用下界写入正常
List<? super Apple> frutis = new ArrayList<Apple>();
fruits.add(new Apple());

对于demo1,因为List<? extends Fruit> 表示 “具有任何从Fruit继承类型的列表”,编译器无法确定List所持有的类型,所以无法安全的向其中添加对象。
对于demo2,list持有的是以Apple为下界,所以编译器可以明确知道添加Apple或Apple的子类是安全的。

5. 注意问题

  • 不能用基本类型实例化类型参数
  • 不能实例化类型变量,例如new T(),T.class等
  • List.class,String[].class,int.class都是允许的,但就List<String>.class和List<?>.class则不合法
  • <Fruit> 和 <Apple>之间没有继承关系,所以List<Fruit>= new ArrayList<Apple>;会报编译错误的,上界和下界为了解决该问题引入。
  • 类型安全和异构容器: 一个集合如果可以预测返回值的类型,则是类型安全的;如果这个集合的键并不是同一个类型的,则称之为异构的。通过包装一个Map,将其中的key保存为Class的,value保存为Object的,可以实现一个异构容器。

参考文档

http://blog.csdn.net/explorers/article/details/454837
http://hongjiang.info/java-generics/
http://www.importnew.com/19740.html

最后编辑于
?著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,100评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,308评论 3 388
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 159,718评论 0 349
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,275评论 1 287
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,376评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,454评论 1 292
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,464评论 3 412
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,248评论 0 269
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,686评论 1 306
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,974评论 2 328
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,150评论 1 342
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,817评论 4 337
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,484评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,140评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,374评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,012评论 2 365
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,041评论 2 351

推荐阅读更多精彩内容

  • Java基础-泛型的约束和局限性 Java中的泛型是一个非常重要知识点,在这里,简单的介绍一下Java泛型的几个注...
    琼珶和予阅读 1,710评论 1 0
  • 一、泛型 什么是泛型 Java 泛型(generics)是 JDK 5 中引入的一个新特性, 泛型提供了编译时类型...
    阿敏其人阅读 8,635评论 0 6
  • 1.什么是泛型 概括地讲,泛型就是我们在定义类、接口或方法的时候将某些引用到的类或接口参数化,这样定义的类、接口、...
    graceLiZi阅读 613评论 0 0
  • 2.简单泛型 -********Java泛型的核心概念:告诉编译器想使用什么类型, 然后编译器帮你处理一切细节 2...
    CodingHou阅读 390评论 0 0
  • 本文大量参考Thinking in java(解析,填充)。 定义:多态算是一种泛化机制,解决了一部分可以应用于多...
    谷歌清洁工阅读 461评论 0 2