java泛型详解
java 泛型的三种定义与使用方法

java 泛型的三种定义与使用方法Java中的泛型主要有三种定义方式,分别是:类型参数、类型参数化类和类型参数化方法。
下面是它们的定义和使用方法:1. 类型参数:类型参数是泛型的主要形式,它允许我们在定义类、接口或方法时指定一个或多个类型参数。
这些类型参数在类、接口或方法的实现中被用作类型占位符,以便在运行时确定实际类型。
定义类型参数的方法是在类、接口或方法的名称后面加上尖括号<>,并在其中指定一个或多个类型参数。
例如:```javapublic class Box<T> {private T content;public Box(T content) {= content;}public T getContent() {return content;}}```在上面的例子中,我们定义了一个名为Box的泛型类,它有一个类型参数T。
我们可以使用任何类型来实例化Box对象,例如Box<Integer>、Box<String>等。
2. 类型参数化类:类型参数化类是将一个类作为泛型参数。
这种定义方式主要用于集合框架中的类,例如List、Set、Map等。
定义类型参数化类的方法是在类名后面加上尖括号<>,并在其中指定一个或多个类型参数。
例如:```javaList<String> list = new ArrayList<>();Set<Integer> set = new HashSet<>();Map<String, Integer> map = new HashMap<>();在上面的例子中,我们定义了三个类型参数化类:List、Set和Map,并使用它们创建了三个不同类型的对象。
这些对象在运行时会自动处理实际类型的匹配。
3. 类型参数化方法:类型参数化方法是在方法中使用泛型。
java泛型定义

java泛型定义
Java泛型是一种在编译时实现类型安全的特性,它允许我们在定义类、接口和方法时使用类型参数。
通过使用泛型,我们可以在编译时发现类型不匹配的错误,从而避免了在运行时出现类型转换异常的情况。
在Java中,泛型是通过在类名或方法名后面添加尖括号(`<`和`>`)来实现的。
泛型类型参数可以是任何有效的Java标识符,通常使用单个大写字母来表示。
例如,`<T>`表示类型参数,`<E>`表示元素类型。
泛型类型参数可以在类、接口和方法中使用。
在类或接口中使用泛型时,我们可以在类或接口名后面添加尖括号,并在括号中指定类型参数。
例如,我们可以定义一个泛型类`List<T>`,其中`T`表示列表中的元素类型。
在方法中使用泛型时,我们可以在方法名和参数列表之间添加尖括号,并在括号中指定类型参数。
例如,我们可以定义一个泛型方法`public <T> T get(int index)`,其中`T`表示方法返回值的类型。
除了定义泛型类和方法外,我们还可以使用泛型通配符来限制类型参数的范围。
通配符可以使用`?`表示,例如`List<?>`表示元素类型未知的列表。
总之,Java泛型是一个非常强大的特性,它可以帮助我们写出更加类型安全、可读性更好的代码。
在使用泛型时,我们应该了解泛型的基本概念和语法,以便正确地定义和使用泛型。
java中泛型参数怎么传参_Java中泛型的使用

java中泛型参数怎么传参_Java中泛型的使用Java中的泛型是指在类、接口、方法的定义中使用一个或多个参数来代表类型,以便在编译时进行类型检查和类型转换,从而增加代码的安全性和可读性。
泛型的使用可以使代码更加灵活、可复用和类型安全。
1.类型参数的使用在Java中,可以在类、接口、方法的定义中使用类型参数,用尖括号<>括起来,放在名称之后。
例如,定义一个泛型类Box,可以存储各种类型的对象:```javapublic class Box<T>private T value;public Box(T value)this.value = value;}public T getValureturn value;}public void setValue(T value)this.value = value;}```在使用泛型类时,可以在创建对象时指定具体的类型参数:```javaBox<Integer> intBox = new Box<Integer>(10);Box<String> strBox = new Box<String>("Hello");```2.类型通配符的使用有时候,我们需要在方法的参数或返回值中使用泛型,但又不确定具体的类型参数。
这时可以使用通配符`?`代表任意类型。
例如,定义一个方法打印数组中的元素:```javapublic static void printArray(Box<?>[] array)for (Box<?> box : array)System.out.println(box.getValue();}```在调用该方法时,可以传入任意类型的Box数组:```javaBox<Integer>[] intArray = new Box<Integer>[10];Box<String>[] strArray = new Box<String>[10];printArray(intArray);printArray(strArray);```3.类型限定的使用有时候,我们需要对泛型参数进行限制,只接受一些类型的参数。
javafan'xing的用法

"Java泛型"是一种在Java编程语言中定义泛型类型的方法。
它是一种用于编写灵活且可重用代码的技术,使得代码能够处理不同的数据类型。
泛型可以让你编写可以处理多种数据类型的代码,而不需要为每种数据类型编写新的代码。
Java泛型的用法主要包括以下步骤:
1. 定义泛型类型:使用尖括号<>来定义泛型类型,并在尖括号中指定泛型类型参数。
例如,List<E>是一个泛型类型,其中E是类型参数。
2. 创建泛型对象:使用泛型类型参数来创建对象。
例如,List<String> list = new ArrayList<String>();创建了一个String类型的List对象。
3. 使用泛型方法:在方法定义中使用泛型类型参数,以便在方法内部使用该参数来操作不同类型的对象。
例如,public static <T> T method(T arg) {return arg;}定义了一个名为method的泛型方法,它接受一个类型为T的参数并返回一个类型为T的对象。
4. 使用通配符:使用通配符来表示未知的泛型类型,以便编写更加灵活的代码。
例如,List<?> list = new ArrayList<String>();表示list 是一个未知类型的List对象,但实际上它是一个String类型的List对象。
总之,Java泛型是一种强大的工具,可以帮助你编写更加灵活、可重用和易于维护的代码。
java中泛型的作用

java中泛型的作用泛型是Java语言的一项强大特性,它允许我们在定义类、接口、方法等时使用类型参数,从而使得代码更加灵活和可复用。
泛型的作用主要体现在以下几个方面:1.类型安全:泛型提供了一种编译时的类型检查机制,可以在编译阶段捕获一些类型错误。
通过使用泛型,我们可以在编译时检查参数类型是否匹配,避免了在运行时可能产生的类型转换异常。
2.代码复用:使用泛型可以编写更加通用的代码,提高代码的可复用性。
通过定义泛型类或泛型方法,我们可以将相似代码逻辑抽象出来,以适用于不同的类型,从而减少重复编写代码的工作量。
3. 集合类的类型安全:泛型集合类(如ArrayList、HashSet等)可以指定存储的元素类型,使得集合中的元素都是指定类型。
这样可以在编译时就避免了将错误类型的对象存入集合的问题,并且在读取集合中的元素时可以直接获取到正确类型的对象,避免了类型转换的繁琐与风险。
4. 简化代码:在没有泛型之前,我们可能需要编写各种类型的容器类,如IntList、StringList等,用于存储不同类型的元素。
而有了泛型,我们可以使用通用的容器类List,通过指定泛型参数的方式来存储不同类型的元素,从而大大简化了代码结构。
5.提高性能:使用泛型可以避免一些不必要的类型转换,从而提高程序的执行效率。
在编译时确定了泛型类型后,编译器会自动插入类型转换的代码,避免了动态类型检查所带来的性能损失。
6.编写更安全的API:通过使用泛型,我们可以将一些类型相关的约定和规范编码到API中,从而可以在编译时强制执行这些约定,提高代码的安全性和可靠性。
7.代码逻辑清晰:泛型可以使代码更具可读性和可维护性。
通过使用泛型,我们可以在编程时直观地表达出代码逻辑的意图,从而使代码更加清晰明了。
总之,泛型是Java语言中一项非常重要的特性,它提供了更高层次的抽象和灵活性,使得我们可以编写更加通用、类型安全且可复用的代码。
通过正确地使用泛型,我们可以提高代码的质量和可维护性,并且更容易适应未来的需求变化。
java泛型语法

java泛型语法Java泛型语法是Java编程语言中的一个重要特性,它允许我们编写更加通用和灵活的代码。
通过使用泛型,我们可以在编译时期检测类型错误,并在运行时期避免类型转换异常。
本文将介绍Java泛型的基本语法和使用方法。
一、泛型的定义和作用泛型是Java中的一种参数化类型,它允许我们在定义类、接口和方法时使用类型参数。
通过使用泛型,我们可以将类型作为参数传递给类、接口和方法,从而实现代码的复用和灵活性。
泛型的作用主要有以下几个方面:1. 类型安全:通过使用泛型,我们可以在编译时期检测类型错误,避免类型转换异常。
2. 代码复用:通过定义泛型类、接口和方法,我们可以实现对多种类型的支持,从而提高代码的复用性。
3. 简化代码:通过使用泛型,我们可以减少冗余的类型转换代码,使代码更加简洁。
4. 提高性能:通过使用泛型,我们可以避免使用Object类型,从而减少了装箱和拆箱的开销,提高了代码的执行效率。
二、泛型的基本语法Java中的泛型通过使用尖括号<>来定义类型参数。
在定义类、接口和方法时,我们可以将类型参数放在尖括号中,并在后续的代码中使用该类型参数。
1. 泛型类的定义:```public class GenericClass<T> {private T data;public T getData() {return data;}public void setData(T data) {this.data = data;}}```在上面的代码中,泛型类GenericClass使用了类型参数T。
我们可以在创建GenericClass对象时指定具体的类型,例如:```GenericClass<String> genericString = new GenericClass<>(); genericString.setData("Hello, World!");String data = genericString.getData();```上面的代码中,我们创建了一个GenericClass对象genericString,并指定了类型参数为String。
java泛型详解-绝对是对泛型方法讲解最详细的,没有之一
java泛型详解-绝对是对泛型⽅法讲解最详细的,没有之⼀1. 概述泛型在java中有很重要的地位,在⾯向对象编程及各种设计模式中有⾮常⼴泛的应⽤。
什么是泛型?为什么要使⽤泛型?泛型,即“参数化类型”。
⼀提到参数,最熟悉的就是定义⽅法时有形参,然后调⽤此⽅法时传递实参。
那么参数化类型怎么理解呢?顾名思义,就是将类型由原来的具体的类型参数化,类似于⽅法中的变量参数,此时类型也定义成参数形式(可以称之为类型形参),然后在使⽤/调⽤时传⼊具体的类型(类型实参)。
泛型的本质是为了参数化类型(在不创建新的类型的情况下,通过泛型指定的不同类型来控制形参具体限制的类型)。
也就是说在泛型使⽤过程中,操作的数据类型被指定为⼀个参数,这种参数类型可以⽤在类、接⼝和⽅法中,分别被称为泛型类、泛型接⼝、泛型⽅法。
2. ⼀个栗⼦⼀个被举了⽆数次的例⼦:1 List arrayList = new ArrayList();2 arrayList.add("aaaa");3 arrayList.add(100);45for(int i = 0; i< arrayList.size();i++){6 String item = (String)arrayList.get(i);7 Log.d("泛型测试","item = " + item);8 }毫⽆疑问,程序的运⾏结果会以崩溃结束:1 ng.ClassCastException: ng.Integer cannot be cast to ng.StringArrayList可以存放任意类型,例⼦中添加了⼀个String类型,添加了⼀个Integer类型,再使⽤时都以String的⽅式使⽤,因此程序崩溃了。
为了解决类似这样的问题(在编译阶段就可以解决),泛型应运⽽⽣。
我们将第⼀⾏声明初始化list的代码更改⼀下,编译器会在编译阶段就能够帮我们发现类似这样的问题。
Java泛型的用法及T.class的获取过程解析
Java泛型的⽤法及T.class的获取过程解析这篇⽂章主要介绍了Java泛型的⽤法及T.class的获取过程解析,⽂中通过⽰例代码介绍的⾮常详细,对⼤家的学习或者⼯作具有⼀定的参考学习价值,需要的朋友可以参考下胡乱总结泛型的四点作⽤:第⼀是泛化,可以拿个T代表任意类型。
但GP是被C++严苛的静态性逼出来的,落到Java、C#这样的花语平原⾥----所有对象除⼏个原始类型外都派⽣于Object,再加上Java的反射功能,Java的Collection库没有范型⼀样过得好好的。
第⼆是泛型 + 反射,原本因为Java的泛型拿不到T.class⽽觉得泛型没⽤,最近才刚刚学到通过反射的API来获取T的Class,后述。
第三是收敛,就是增加了类型安全,减少了强制类型转换的代码。
这点倒是Java Collection历来的弱项。
第四是可以在编译期搞很多东西,⽐如MetaProgramming。
但除⾮能完全封闭于框架内部,框架的使⽤者和扩展者都不⽤学习这些东西的⽤法,否则那就是⾃绝于⼈民的票房毒药。
C++的MetaProgramming好厉害吧,但对⽐⼀下Python拿Meta Programming⽣造⼀个Class出来的简便语法,就明⽩什么才是真正的叫好⼜叫座。
所以,作为⼀个架构设计师,应该使⽤上述的第2,3项⽤法,在框架类⾥配合使⽤反射和泛型,使得框架的能⼒更强;同时采⽤收敛特性,本着对⼈民负责的精神,⽤泛型使框架更加类型安全,更少强制类型转换。
擦拭法避免了Java的流⾎分裂:⼤家经常骂Java GP的擦拭法实现,但我觉得多亏于它的中庸特性---如果你⽤就是范型,不⽤就是普通Object,避免了Java阵营⼜要经历⼀场to be or not to be的分裂。
最⼤的例⼦莫过Java 5的Collection 框架,⽐如有些同学坚持认为⾃⼰不会⽩痴到类型出错,⽽且难以忍受每个定义的地⽅都要带⼀个泛型定义List〈Book〉,不⽤强制类型转换所省下的代码还不够N处定义花的(对了,java⾥⾯还没有tyepdef.....),因此对范型⼗分不感冒,这时就要齐齐感谢这个搽拭法让你依然可以对⼀个泛型框架保持⾮泛型的⽤法了...<<⼲货来了>>通过反射获得 T.class:abstract public class BaseHibernateEntityDao<T> extends HibernateDaoSupport {private Class<T> entityClass;public BaseHibernateEntityDao() {entityClass =(Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];} public T get(Serializable id) { T o = (T) getHibernateTemplate().get(entityClass, id); return o; } }重点就是这句话:Class<T> entityClass = (Class<T>) ((ParameterizedType) getClass().getGenericSuperclass()).getActualTypeArguments()[0];解释:1. public Type getGenericSuperclass()⽤来返回表⽰当前Class 所表⽰的实体(类、接⼝、基本类型或 void)的直接超类的Type。
网络程序设计java第05章泛型和枚举
参数化测试
在单元测试中,可以使用泛型来 参数化测试用例,对不同的输入 类型进行测试。
枚举在常量定义和类型选择中的应用场景
常量定义
使用枚举可以定义一组有限的常量值, 每个常量值都有明确的含义和用途。
类型选择
通过枚举,可以在程序中选择一个固 定的数据类型,例如表示星期的枚举、 表示月份的枚举等。
状态管理
分隔。
枚举常量
每个枚举常量都是该枚举类型 的一个实例,可以使用默认构
造函数创建枚举常量对象。
枚举方法
可以在枚举类型中定义方法, 这些方法将应用于所有的枚举
常量。
枚举与继承
枚举继承ห้องสมุดไป่ตู้
枚举类型可以继承另一个枚举类型,继承的枚举常量将自动 包含在子类中。
覆盖方法
可以在子枚举类型中覆盖父类中的方法,并为其提供新的实 现。
Java集合框架中的类如`ArrayList`, `HashSet`等支持 泛型,可以创建特定类型的集合。
集合的元素类型限制
使用泛型可以限制集合中元素的类型,减少类型转换 和运行时异常。
泛型通配符
使用通配符`?`表示未知类型,例如`List<?>`表示任意 类型的列表。
泛型与继承
泛型继承
子类可以继承父类的泛型类型,例如`class SpecialBox<T> extends Box<T>`。
实现泛型接口时需要指定具体的类型参数,例如`class StringGenerator implements Generator<String>`。
03
泛型的类型参数
类型参数的命名规则
01
类型参数的名称必须以大写字母开头,遵循驼峰命名法。
java 泛型用法实例
java 泛型用法实例Java 泛型是一种将类型参数化的方法,通过它可以在编译时检测到类型不匹配的错误。
泛型的核心思想是参数化类型,实现了代码的重用性和类型的安全性。
下面我将通过一些实例来介绍 Java 泛型的用法。
1. 泛型类泛型类是指在定义类时使用类型参数来代替真实类型。
例如,我们可以定义如下的一个泛型类来表示一个有序对(Ordered Pair):```public class OrderedPair<K, V> {private K key;private V value;public OrderedPair(K key, V value) {this.key = key;this.value = value;}public K getKey() {return key;}public V getValue() {return value;}}```在上面的例子中,`K` 和 `V` 是类型参数,用来代替真实的类型。
在使用该类时,我们可以指定实际的类型来创建对象:```OrderedPair<Integer, String> p1 = new OrderedPair<Integer, String>(1, "One");OrderedPair<String, Integer> p2 = new OrderedPair<String, Integer>("Two", 2);```2. 泛型方法泛型方法是指在定义方法时使用类型参数来代替方法中的参数类型。
例如,我们可以定义如下的一个泛型方法来交换数组中的两个元素:```public static <T> void swap(T[] array, int i, int j) {T temp = array[i];array[i] = array[j];array[j] = temp;}```在上面的例子中,`<T>` 表示该方法是一个泛型方法,类型参数 `T` 用来代替真实的类型。
- 1、下载文档前请自行甄别文档内容的完整性,平台不提供额外的编辑、内容补充、找答案等附加服务。
- 2、"仅部分预览"的文档,不可在线预览部分如存在完整性等问题,可反馈申请退款(可完整预览的文档不适用该条件!)。
- 3、如文档侵犯您的权益,请联系客服反馈,我们会尽快为您处理(人工客服工作时间:9:00-18:30)。
java泛型详解泛型(Generic type 或者generics)是对 Java 语言的类型系统的一种扩展,以支持创建可以按类型进行参数化的类。
可以把类型参数看作是使用参数化类型时指定的类型的一个占位符,就像方法的形式参数是运行时传递的值的占位符一样。
可以在集合框架(Collection framework)中看到泛型的动机。
例如,Map类允许您向一个Map添加任意类的对象,即使最常见的情况是在给定映射(map)中保存某个特定类型(比如String)的对象。
因为Map.get()被定义为返回Object,所以一般必须将Map.get()的结果强制类型转换为期望的类型,如下面的代码所示:Map m = new HashMap();m.put("key", "blarg");String s = (String) m.get("key");要让程序通过编译,必须将get()的结果强制类型转换为String,并且希望结果真的是一个String。
但是有可能某人已经在该映射中保存了不是String的东西,这样的话,上面的代码将会抛出ClassCastException。
理想情况下,您可能会得出这样一个观点,即m是一个Map,它将String键映射到String值。
这可以让您消除代码中的强制类型转换,同时获得一个附加的类型检查层,该检查层可以防止有人将错误类型的键或值保存在集合中。
这就是泛型所做的工作。
泛型的好处Java 语言中引入泛型是一个较大的功能增强。
不仅语言、类型系统和编译器有了较大的变化,以支持泛型,而且类库也进行了大翻修,所以许多重要的类,比如集合框架,都已经成为泛型化的了。
这带来了很多好处:· 类型安全。
泛型的主要目标是提高 Java 程序的类型安全。
通过知道使用泛型定义的变量的类型限制,编译器可以在一个高得多的程度上验证类型假设。
没有泛型,这些假设就只存在于程序员的头脑中(或者如果幸运的话,还存在于代码注释中)。
Java 程序中的一种流行技术是定义这样的集合,即它的元素或键是公共类型的,比如“Str ing列表”或者“String到String的映射”。
通过在变量声明中捕获这一附加的类型信息,泛型允许编译器实施这些附加的类型约束。
类型错误现在就可以在编译时被捕获了,而不是在运行时当作 ClassCastException展示出来。
将类型检查从运行时挪到编译时有助于您更容易找到错误,并可提高程序的可靠性。
· 消除强制类型转换。
泛型的一个附带好处是,消除源代码中的许多强制类型转换。
这使得代码更加可读,并且减少了出错机会。
尽管减少强制类型转换可以降低使用泛型类的代码的罗嗦程度,但是声明泛型变量会带来相应的罗嗦。
比较下面两个代码例子。
该代码不使用泛型:List li = new ArrayList();li.put(new Integer(3));Integer i = (Integer) li.get(0);该代码使用泛型:List<Integer> li = new ArrayList<Integer>();li.put(new Integer(3));Integer i = li.get(0);在简单的程序中使用一次泛型变量不会降低罗嗦程度。
但是对于多次使用泛型变量的大型程序来说,则可以累积起来降低罗嗦程度。
· 潜在的性能收益。
泛型为较大的优化带来可能。
在泛型的初始实现中,编译器将强制类型转换(没有泛型的话,程序员会指定这些强制类型转换)插入生成的字节码中。
但是更多类型信息可用于编译器这一事实,为未来版本的JVM 的优化带来可能。
由于泛型的实现方式,支持泛型(几乎)不需要JVM 或类文件更改。
所有工作都在编译器中完成,编译器生成类似于没有泛型(和强制类型转换)时所写的代码,只是更能确保类型安全而已。
泛型用法的例子泛型的许多最佳例子都来自集合框架,因为泛型让您在保存在集合中的元素上指定类型约束。
考虑这个使用Map类的例子,其中涉及一定程度的优化,即Map.get()返回的结果将确实是一个String:Map m = new HashMap();m.put("key", "blarg");String s = (String) m.get("key");如果有人已经在映射中放置了不是String的其他东西,上面的代码将会抛出ClassCastException。
泛型允许您表达这样的类型约束,即m是一个将String 键映射到String值的Map。
这可以消除代码中的强制类型转换,同时获得一个附加的类型检查层,这个检查层可以防止有人将错误类型的键或值保存在集合中。
下面的代码示例展示了 JDK 5.0 中集合框架中的Map接口的定义的一部分:public interface Map<K, V> {public void put(K key, V value);public V get(K key);}注意该接口的两个附加物:* 类型参数 K 和 V 在类级别的规格说明,表示在声明一个 Map 类型的变量时指定的类型的占位符。
* 在 get()、put() 和其他方法的方法签名中使用的 K 和 V。
为了赢得使用泛型的好处,必须在定义或实例化Map类型的变量时为K和V提供具体的值。
以一种相对直观的方式做这件事:Map<String, String> m = new HashMap<String, String>();m.put("key", "blarg");String s = m.get("key");当使用Map的泛型化版本时,您不再需要将Map.get()的结果强制类型转换为String,因为编译器知道get()将返回一个String。
在使用泛型的版本中并没有减少键盘录入;实际上,比使用强制类型转换的版本需要做更多键入。
使用泛型只是带来了附加的类型安全。
因为编译器知道关于您将放进Map中的键和值的类型的更多信息,所以类型检查从执行时挪到了编译时,这会提高可靠性并加快开发速度。
向后兼容在 Java 语言中引入泛型的一个重要目标就是维护向后兼容。
尽管 JDK 5.0 的标准类库中的许多类,比如集合框架,都已经泛型化了,但是使用集合类(比如HashMap和ArrayList)的现有代码将继续不加修改地在 JDK 5.0 中工作。
当然,没有利用泛型的现有代码将不会赢得泛型的类型安全好处。
类型参数在定义泛型类或声明泛型类的变量时,使用尖括号来指定形式类型参数。
形式类型参数与实际类型参数之间的关系类似于形式方法参数与实际方法参数之间的关系,只是类型参数表示类型,而不是表示值。
泛型类中的类型参数几乎可以用于任何可以使用类名的地方。
例如,下面是java.util.Map接口的定义的摘录:public interface Map<K, V> {public void put(K key, V value);public V get(K key);}Map接口是由两个类型参数化的,这两个类型是键类型K和值类型V。
(不使用泛型)将会接受或返回Object的方法现在在它们的方法签名中使用K或V,指示附加的类型约束位于Map的规格说明之下。
当声明或者实例化一个泛型的对象时,必须指定类型参数的值:Map<String, String> map = new HashMap<String, String>();注意,在本例中,必须指定两次类型参数。
一次是在声明变量map的类型时,另一次是在选择HashMap类的参数化以便可以实例化正确类型的一个实例时。
编译器在遇到一个Map<String, String>类型的变量时,知道K和V现在被绑定为String,因此它知道在这样的变量上调用Map.get()将会得到String类型。
除了异常类型、枚举或匿名内部类以外,任何类都可以具有类型参数。
命名类型参数推荐的命名约定是使用大写的单个字母名称作为类型参数。
这与C++ 约定有所不同(参阅附录 A:与 C++ 模板的比较),并反映了大多数泛型类将具有少量类型参数的假定。
对于常见的泛型模式,推荐的名称是:* K ——键,比如映射的键。
* V ——值,比如 List 和 Set 的内容,或者 Map 中的值。
* E ——异常类。
* T ——泛型。
泛型不是协变的关于泛型的混淆,一个常见的来源就是假设它们像数组一样是协变的。
其实它们不是协变的。
List<Object>不是List<String>的父类型。
如果 A 扩展 B,那么 A 的数组也是 B 的数组,并且完全可以在需要B[]的地方使用A[]:Integer[] intArray = new Integer[10];Number[] numberArray = intArray;上面的代码是有效的,因为一个Integer是一个Number,因而一个Integer数组是一个Number数组。
但是对于泛型来说则不然。
下面的代码是无效的:List<Integer> intList = new ArrayList<Integer>();List<Number> numberList = intList; // invalid最初,大多数 Java 程序员觉得这缺少协变很烦人,或者甚至是“坏的(broken)”,但是之所以这样有一个很好的原因。
如果可以将List<Integer>赋给List<Number>,下面的代码就会违背泛型应该提供的类型安全:List<Integer> intList = new ArrayList<Integer>();List<Number> numberList = intList; // invalidnumberList.add(new Float(3.1415));因为intList和numberList都是有别名的,如果允许的话,上面的代码就会让您将不是Integers的东西放进intList中。
但是,正如下一屏将会看到的,您有一个更加灵活的方式来定义泛型。
package com.ibm.course.generics;import java.util.ArrayList;import java.util.List;public class GenericsExample {public static void main(String[] args) {Integer[] integer = new Integer[5];Number[] number = integer;System.out.println(number[0]);// nullnumber[0] = new Float(7.65);System.out.println(number[0]);System.out.println(integer[0]);List<Integer> list = new ArrayList<Integer>();// Type mismatch: cannot convert from List<Integer> to List<Number> // List<Number> listObj = list;}}List<Number> listObj = list;导致编译错误:Type mismatch: cannot convert from List<Integer> to List<Number>而System.out.println(number[0]);和System.out.println(integer[0]);导致运行时异常:Exception in thread "main" ng.ArrayStoreException:ng.Floatatcom.ibm.course.generics.GenericsExample.main(GenericsExample.java:15)类型通配符假设您具有该方法:void printList(List l) {for (Object o : l)System.out.println(o);}上面的代码在 JDK 5.0 上编译通过,但是如果试图用List<Integer>调用它,则会得到警告。