Java - 泛型通配符 ? 与 T 的区别

简介: Java - 泛型通配符 ? 与 T 的区别

之前对Java的泛型不太熟悉,没怎么搞明白通配符 ? 的使用,以及 ? 与 T 的区别,导致一直懵懵懂懂,也不太敢用。

网上找了一大堆解释文章,都没有怎么讲清晰,最后发现是Java官方的教程《Lesson: Generics》,把泛型和通配符的问题讲得非常清楚。

List , 这个 T 是一个形参,可以理解为一个占位符,被使用时,会在程序运行的时候替换成具体的类型,比如替换成String,Integer之类的。

List, 这个 ? 是一个实参,这是Java定义的一种特殊类型,比Object更特殊,就像一个影子。比如List和List是没有父子关系的,这是两个类型,List类型和List类型;但是List 是 List的父类。

用数学集合的语言来表述,? 表示了集合【所有Java类型,String,Integer等系统定义的,或者用户定义的Foo等类型】这个整体;而 T 表示了集合【所有Java类型,String,Integer等系统定义的,或者用户定义的Foo等类型】中的一个成员。

正是因为 ? 是个集合,T 是集合中的一个成员,导致我们很容易混淆这两者到底有什么区别。在一些情况下,这两者确实是可以相互替换的,但是另一种情况下就不行了。

?表示了任何的一种类型,那 List 岂不是可以包含 String 和 Integer,但这又和Java的类型系统矛盾了,List里面只能放一种类型。于是乎,对于 List list 是不可能进行 list.add(1) 的,不能对它进行写操作,除了可以 list.add(null)

而对于 List, 却是可以进行写操作的


public static <T extends Number> void addTExtend(List<T> list, T e){
    list.add(e);
}

那List到底有什么意义呢?在我不需要处理数组里的元素的时候,写代码更方便简单。如下两个方法的功能是一样的,但是使用List简单些。


public static void setNullWildcard(List<?> list){
    list = null;
}
public static <T> void setNullT(List<T> list){
    list = null;
}

List里面的元素,如果你取出来,会都被转化为Object(因为Object是这个集合里所有元素的父),如果你只需要用到这个集合最顶层的父元素的方法,比如List就是Object的方法,List 就是 Number 的方法,那你也可以使用List以及List 来简化代码的书写

public static <T> void printList(List<T> list){
    for(T e: list){
        System.out.print(e + " ");
    }
    System.out.println("");
}
public static void printListWildCard(List<?> list){
    for(Object e: list){
        System.out.print(e + " ");
    }
    System.out.println("");
}

如上两个方法,功能都是一样的,只是用 List 更简便。

总结 ? 相对于 T 的第一个区别:不关心List里面的元素,或者只需要用到List里面元素的最顶层父元素的方法的时候,可以用List来简化代码的书写。

上面说到了 List,如下代码,使用 ? 还是 T ,都可以实现同样的功能

public static void printListWildCardExtend(List<? extends Number> list){
    for(Number e: list){
        System.out.print(e + " ");
    }
    System.out.println("");
}
public static <T extends Number> void printListTExtend(List<T> list){
    for(Number e: list){
        System.out.print(e + " ");
    }
    System.out.println("");
} 

两者都可以通过extends来限定一个类型的子集,但是 T 可以 List 即限定为多重继承的,? 却不可以


public static <T extends Number & ExtendInterface> void printListTExtend(List<T> list){
    for(Number e: list){
        System.out.print(e + " ");
    }
    System.out.println("");
}

无法实现 void printListWildCardExtend(List list)

总结 ? 与 T 的第二个区别:使用extends限定类型子集的时候,?不能多重继承,T 可以


public static void printListSuperNumber(List<? super Integer> list){
    for(Object e: list) {
        System.out.print(e + " ");
    }
    System.out.println("");
}

?是可以限定父集的,但是 T 是做不到这一点的。

总结?与 T的第三个区别:使用super限定父集的时候,? 可以, T 不可以


目录
相关文章
|
21天前
|
Java 程序员
Java社招面试题:& 和 && 的区别,HR的套路险些让我翻车!
小米,29岁程序员,分享了一次面试经历,详细解析了Java中&和&&的区别及应用场景,展示了扎实的基础知识和良好的应变能力,最终成功获得Offer。
53 14
|
16天前
|
Java
java中面向过程和面向对象区别?
java中面向过程和面向对象区别?
19 1
|
26天前
|
存储 缓存 安全
java 中操作字符串都有哪些类,它们之间有什么区别
Java中操作字符串的类主要有String、StringBuilder和StringBuffer。String是不可变的,每次操作都会生成新对象;StringBuilder和StringBuffer都是可变的,但StringBuilder是非线程安全的,而StringBuffer是线程安全的,因此性能略低。
44 8
|
1月前
|
Java
Java代码解释++i和i++的五个主要区别
本文介绍了前缀递增(++i)和后缀递增(i++)的区别。两者在独立语句中无差异,但在赋值表达式中,i++ 返回原值,++i 返回新值;在复杂表达式中计算顺序不同;在循环中虽结果相同但使用方式有别。最后通过 `Counter` 类模拟了两者的内部实现原理。
Java代码解释++i和i++的五个主要区别
|
2月前
|
Java
通过Java代码解释成员变量(实例变量)和局部变量的区别
本文通过一个Java示例,详细解释了成员变量(实例变量)和局部变量的区别。成员变量属于类的一部分,每个对象有独立的副本;局部变量则在方法或代码块内部声明,作用范围仅限于此。示例代码展示了如何在类中声明和使用这两种变量。
|
2月前
|
Java API
[Java]泛型
本文详细介绍了Java泛型的相关概念和使用方法,包括类型判断、继承泛型类或实现泛型接口、泛型通配符、泛型方法、泛型上下边界、静态方法中使用泛型等内容。作者通过多个示例和测试代码,深入浅出地解释了泛型的原理和应用场景,帮助读者更好地理解和掌握Java泛型的使用技巧。文章还探讨了一些常见的疑惑和误区,如泛型擦除和基本数据类型数组的使用限制。最后,作者强调了泛型在实际开发中的重要性和应用价值。
41 0
[Java]泛型
|
2月前
|
存储 安全 Java
🌱Java零基础 - 泛型详解
【10月更文挑战第7天】本文收录于「滚雪球学Java」专栏,专业攻坚指数级提升,希望能够助你一臂之力,帮你早日登顶实现财富自由🚀;同时,欢迎大家关注&&收藏&&订阅!持续更新中,up!up!up!!
19 1
|
2月前
|
Java
Java代码解释静态代理和动态代理的区别
### 静态代理与动态代理简介 **静态代理**:代理类在编译时已确定,目标对象和代理对象都实现同一接口。代理类包含对目标对象的引用,并在调用方法时添加额外操作。 **动态代理**:利用Java反射机制在运行时生成代理类,更加灵活。通过`Proxy`类和`InvocationHandler`接口实现,无需提前知道接口的具体实现细节。 示例代码展示了两种代理方式的实现,静态代理需要手动创建代理对象,而动态代理通过反射机制自动创建。
|
2月前
|
缓存 算法 Java
Java 中线程和纤程Fiber的区别是什么?
【10月更文挑战第14天】
93 0
|
6月前
|
Java API 容器
Java泛型的继承和通配符
Java泛型的继承和通配符
40 1
下一篇
DataWorks