Java 泛型

什么是泛型

泛型,简单来讲就是在定义类、接口或方法的时候,将数据类型参数化,由使用这些类/接口/方法的一方传入所需要的数据类型,并有编译器进行类型安全检查和强制转换。例如我们常用的List<Integer>就是泛型的一种用法,我们通过<Integer>指定了这个List的类型只能时Integer。

泛型的好处

  1. 提高了代码的重用率,一段代码可以被不同数据类型的对象所重用
  2. 提供了编译阶段的类型安全检查,尤其是对集合类型而言。如下面代码中,list作为一个数据容器,可以存放任何类型,但若取出数据的时候没有做类型检查的话,就会报“java.lang.ClassCastException: java.lang.String cannot be cast to java.lang.Integer”数据类型转换异常的错,这种错误若完全靠程序员自己避免的话,是一件十分危险和低效的事情。
List list = new ArrayList();
list.add(1);
list.add("test");
list.add(1.1);
for (int i = 0; i < list.size(); i++) {
   int item = (int)list.get(i);
   System.out.println(item);
}

但若使用泛型就可以避免这种事情,不符合类型规定的数据就会被编译器发现并报错

List<Integer> list = new ArrayList();
list.add(1);
list.add("test"); // 在编译阶段就会报错

泛型的实现

简单概述一下,Java的泛型是编译器层面的技术,编译器在生成Java字节码时并不会记录泛型中的类型信息,只会保留原始类型(raw type),而仅在Class的实例中包含了类型参数的定义信息,之后当调用泛型对象的时候会再进行数据类型的强制转换。编译器去除类型参数信息这个过程就称为类型擦除。
下面的代码可以证明在Runtime中不管是Integer还是String的信息都被擦除了,对编译器来讲两个list都是ArrayList。

List<Integer> listInteger = new ArrayList();
List<String> listString = new ArrayList<>();
System.out.println(listInteger.getClass().equals(listString.getClass()));//true
System.out.println(listInteger.getClass());//class java.util.ArrayList
System.out.println(listString.getClass());//class java.util.ArrayList

泛型的用法

在Java中,可以在类、接口或方法的定义后面用尖括号(<>)和类型参数表示泛型,形如<T>。其中类型参数起着占位符的作用,指示在运行时为类分配类型。根据实际需要,可以有一个或多个类型参数(如GenericContainer<T, E>)。
值得注意的是,在传入类型实参的时候,数据类型必须为包装类,不能为简单类型。
习惯上是单个大写字母表示类型参数,并有以下的标准类型参数:

  • E:元素
  • K:键
  • N:数字
  • T:类型
  • V:值
  • S、U、V 等:多参数情况中的第 2、3、4 个类型

1、泛型类

泛型类型用于类的定义中,被称为泛型类。最典型的就是各种容器类,如:List、Set、Map。通过代码直观地看一下。
这是一个泛型类GenericContainer。

public class GenericContainer<T> {
    private T obj;

    public GenericContainer(){
    }
    
    public GenericContainer(T t){
        obj = t;
    }
    
    public T getObj() {
        return obj;
    }
    
    public void setObj(T t) {
        obj = t;
    }
}

当声明一个泛型实例的时候,若传入类型参数,则编译器会对之后对数据做类型检查,并对不兼容的类型抛出错误。

GenericContainer<Integer> integerContainer = new GenericContainer<>(1);
GenericContainer<String> stringContainer = new GenericContainer<>("string");
System.out.println(integerContainer.getObj()); //1
System.out.println(stringContainer.getObj()); //string
// Error:(18, 33) java: 不兼容的类型: java.lang.String无法转换为java.lang.Integer
integerContainer.setObj("test");

当没有传入类型参数的时候,泛型类也是可以被初始化的,且此时编译器不会被传入的数据进行类型检查,这样也就失去类泛型的好处,在强制类型转换的时候会抛出ClassCastException异常

GenericContainer container = new GenericContainer();
container.setObj(1);
System.out.println(container.getObj()); //1
container.setObj("string");
System.out.println(stringContainer.getObj()); //string
//抛出java.lang.ClassCastException异常
Integer obj = (Integer) container.getObj();

2、泛型接口

泛型类型用于接口的定义中时,此接口成为泛型接口,它的定义和用法与泛型类基本相同。泛型的实现类必须申明泛型(即类命后的<T>不可丢)或者传入类型实参。代码如下:

public interface GenericInterface<T> {
    public T get();
}
// 不做类型限制,由实例指定参数类型
public class GenericInImpl<T> implements GenericInterface<T> {
    @Override
    public String get() { 
        return null;
    }
}

// 实现类已经限制了类型
public class GenericInImplTwo implements GenericInterface<String> {
    @Override
    public T get() {//此处的T应该替换成已经指定的String类型
        return null;
    }
}

3、泛型方法

形式如下的function称为泛型方法。可以是参数类型为T也可以是返回值为T。
“public”和函数的返回值之间的<T>不可丢,这是申明该方法是泛型方法的标志。这种方法多用于编写工具,比如Object2Json这种通用工具方法。

public <T> T returnT(T data){
    T obj = (T)new Object();
     return obj;
}

泛型方法容易和泛型类/泛型接口中的方法混淆。值得注意的是,泛型类/泛型接口中的方法并不是泛型方法,只是需要用到类定义中的类型参数的普通方法而已。泛型类/泛型接口中也可以定义泛型方法,此时该方法的类型参数是独立于类/接口的,也就是说调用该方法的时候可以传入与类/接口定义中相同的类型,也可以是不同的类型。以代码为例:

public class GenericContainer<T> {
    private T obj;

    public GenericContainer(T t){
        obj = t;
    }
    public void printClass1(T t){
        System.out.println(t.getClass());
    }
    public <T> void printClass2(T t){
        System.out.println(t.getClass());
    }

    public <E> void printClass3(E e){
        System.out.println(e.getClass());
    }
}

从下面代码的执行结果中我们可以看到用<T>申明的泛型方法是不受泛型类的类型约束的,而是根据调用时传入的参数类型约束。

GenericContainer<String> container = new GenericContainer<>("string");
// Error:(27, 31) java: 不兼容的类型: int无法转换为java.lang.String
container.printClass1(1);
// class java.lang.Integer
container.printClass2(1);
// class java.lang.Double
container.printClass3(1.1);

4、泛型通配符

我们知道,对象或者数组是可以向上转型的,如下面代码是可以通过编译和运行的。

Number number = new Integer();
Number[] numbers = new Integer[10];

但泛型容器是不支持的,比如下面的代码就无法通过编译,尽管Integer是Number的子类,但ArrayList<Integer>并不是ArrayList<Number>的子类。

// Compile Error: incompatible types:
 ArrayList<Number> numberList =  new ArrayList<Integer>();

此时我们引入一个通配符的概念,符号标志为"?",代表着某种类型,但不知道具体但类型。值得注意的是,用了?通配符后,该数组就不再被允许添加,只能查询删除等操作,因为此时我们不知道该数组的具体类型,往里面进行添加数据是一种非常不安全的操作。从通配符标志的数组中取出来的数据需要显式地做数据类型转换。通配符可以通过“extends”、“super”界定范围,其中利用“super”范围的数组可以添加数据。详见第5小节。

// 合法,但不允许add操作
ArrayList<?> numbers =  new ArrayList<Integer>();
List<?> numberList = Arrays.asList(1);
Integer number = (Integer)numberList.get(0);
System.out.println(number);// 1

5、PECS

PECS即“Producer Extends,Consumer Super”的简称,“extends”和“super”都是用来进行泛型限定的修饰符。
“extends‘代表的是上限,“T extends Number”表示可以接收Number类型或者Number的子类型对象。也可以和通配符一起使用,"? extends Number"也代表可以接收Number类型或者Number的子类型对象。“Producer Extends”的含义是若参数化类型表示一个生产者,就使用<? extends T>。举个具体的例子

// 合法
ArrayList<? extends  Number> numberList =  new ArrayList<Integer>();
// 不合法,无法通过编译
ArrayList<? extends  Number> numberList2 =  new ArrayList<String>();

在以上代码中,numberList被限制为了可以接收Number类型或者Number的子类型对象的数组,如果给它赋予整数类型的数组是合法的,但赋予字符串类型的数组就无法通过编译,因为超出了界限。同时,还有一点非常重要,就是numberList只能作为数据的生产者,即提供数据给别人消费,但不能消费数据。从数组的角度来讲,就是可以获取numberList中的数据,但不能往里面添加,从而避免了下面这种不安全的情况

ArrayList<? extends  Number> numberList =  new ArrayList<Integer>();
numberList.add(1);
numberList.add(1.1);
Integer number = (Double)numberList.get(1);// 类型转换异常

“super‘代表的是下限,“T super Integer”:可以接收Integer类型或者Integer的父类型对象。同样super也可以和通配符一起使用。也就是说我们不知道这个T或者?符号代表着什么类,但这个类一定是Integer的父类。“Consumer Super”的含义是若参数化类型表示一个数据的消费者者,就使用<? super T>,在数组上的表现是你可以往里面add数据。

PECS之外,其实还有一个约定,就是如果一个数组即是生产者又是消费者,那么它一个用List<T>,而不是List<? extends T>或List<? super T>。

总结

  1. 泛型的类型参数只能是包装类(包括自定义类),不能是简单类型;
  2. 同一种泛型可以对应多个版本(因为参数类型是不确定的),不同版本的泛型类实例是不兼容的;
  3. 泛型的类型参数可以有多个;
  4. 泛型的参数类型可以使用extends、super语句,用于约束类型的界限
  5. 泛型的参数类型还可以是通配符类型。例如Class<?> classType = Class.forName("java.lang.String")
  6. Producer Extends,Consumer Super

参考
JAVA泛型实现原理
泛型的内部原理
泛型:工作原理及其重要性

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

推荐阅读更多精彩内容

  • 我们知道,使用变量之前要定义,定义一个变量时必须要指明它的数据类型,什么样的数据类型赋给什么样的值。 假如我们现在...
    今晚打肉山阅读 970评论 0 1
  • 一. 为什么需要泛型 先来说说一个简单的实例。 这个例子很简单,有一个仓库Store,里面储存了一个数据data,...
    wo883721阅读 765评论 0 3
  • Java 泛型是 Java 5 引入的一个重要特性,相信大多数 Java 开发者都对此不陌生,但是泛型背后的实现原...
    JohnnyShieh阅读 2,055评论 6 37
  • 参数类型的好处 在 Java 引入泛型之前,泛型程序设计是用继承实现的。ArrayList 类只维护一个 Obje...
    杰哥长得帅阅读 870评论 0 3
  • 开发人员在使用泛型的时候,很容易根据自己的直觉而犯一些错误。比如一个方法如果接收List作为形式参数,那么如果尝试...
    时待吾阅读 1,040评论 0 3