Java泛型总结
Table of Contents
一.概述
1.什么是Java泛型
泛型,即“参数化类型”。就是将类型由原来具体的类型, 比如Food food
这个Food
类型改为参数化, 由外部调用的时候来指定。
2.Java泛型的意义
使用泛型的意义在于:
- a.封装及代码复用, 适用于多种数据类型执行相同的代码(比如Request<T>, Response<T>), 这样就可以通过对外开放相同的接口来完成对一组类的操作
- b.在编译时期进行类型安全检查 (比如ArrayList<T>)
- c.控制数据安全访问, 即PECS法则。
二.泛型类
泛型类型用于类的定义中,被称为泛型类。
// 此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型
class GenericClass<T> {
T data;
public GenericClass() {
}
public GenericClass(T data) {
this.data = data;
}
public void setData(T data) {
this.data = data;
}
public T getData() {
return data;
}
}
// 在实例化泛型类时,传入的实参类型(说的是右侧的<String>)需与泛型的类型参数类型(说的是左侧的<String>)相同。
// 右侧的可以推断出来, 因此可以省略。但是要写的话,一定要相同。
GenericClass<String> s = new GenericClass<String>("123");
// 在使用泛型的时候如果传入泛型实参,则会根据传入的泛型实参做相应的限制,此时泛型才会起到本应起到的限制作用。
// 如果不传入泛型类型实参的话,在泛型类中使用泛型的方法或成员变量定义的类型可以为任何的类型。
GenericClass generic = new GenericClass("111111");
GenericClass generic1 = new GenericClass(4444);
GenericClass generic2 = new GenericClass(55.55);
GenericClass generic3 = new GenericClass(false);
三.泛型接口
泛型类型用于接口的定义中,被称为泛型类。
interface Generator<T> {
public T next();
}
泛型接口的实现方式有两种:
// 实现泛型接口的方式1
// 接口中未传入泛型实参,需将泛型的声明也一起加到类中, 也就是在类的声明中写上<T>
class FruitGenerator<T> implements Generator<T> {
@Override
public T next() {
return null;
}
}
// 实现泛型接口的方式2
// 在接口中指定泛型实参, 这样类中的泛型就规定好了, 所有使用泛型的地方都要替换成传入的实参类型, 这里是String
class FoodGenerator implements Generator<String> {
@Override
public String next() {
return null;
}
}
四.泛型方法
泛型类,是在实例化类的时候指明泛型的具体类型;泛型方法,是在调用方法的时候指明泛型的具体类型。
1.泛型方法基本介绍
/**
* 泛型方法的基本介绍
* @param tClass 传入的泛型实参
* @return T 返回值为T类型
* 说明:
* 1)public 与 返回值中间<T>非常重要,可以理解为声明此方法为泛型方法。
* 2)只有声明了<T>的方法才是泛型方法,泛型类中的使用了泛型的成员方法并不是泛型方法。
* 3)<T>表明该方法将使用泛型类型T,此时才可以在方法中使用泛型类型T。
* 4)与泛型类的定义一样,此处T可以随便写为任意标识,常见的如T、E、K、V等形式的参数常用于表示泛型。
*/
public <T> T genericMethod(Class<T> tClass)throws InstantiationException,
IllegalAccessException{
T instance = tClass.newInstance();
return instance;
}
Object obj = genericMethod(Class.forName("com.test.test"));
2.泛型方法的说明
class GenericMethodTest {
// 这个类是个泛型类,在上面已经介绍过
public class Generic<T> {
private T key;
public Generic(T key) {
this.key = key;
}
// 这个方法虽然使用了泛型,但是这并不是一个泛型方法。
// 这只是类中一个普通的成员方法,只不过他的返回值是在声明泛型类已经声明过的泛型。
// 所以在这个方法中才可以继续使用 T 这个泛型。
public T getKey() {
return key;
}
/**
* 这个方法显然是有问题的,在编译器会给我们提示这样的错误信息"cannot resolve symbol E"
* 因为在类的声明中并未声明泛型E,所以在使用E做形参和返回值类型时,编译器会无法识别。
*
* public E setKey(E key){
* this.key = keu
* }
*/
}
/**
* 这才是一个真正的泛型方法。
* 首先在public与返回值之间的<T>必不可少,这表明这是一个泛型方法,并且声明了一个泛型T
* 这个 T 可以出现在这个泛型方法的任意位置.
* 泛型的数量也可以为任意多个
* 如:
* public <T,K> K showKeyName(Generic<T> container){
* ...
* }
*/
public <T> T showKeyName(Generic<T> container) {
T test = container.getKey();
return test;
}
/**
* 这也不是一个泛型方法,这就是一个普通的方法,只是使用了Generic<Number>这个泛型类做形参而已。
*/
public void showKeyValue1(Generic<Number> obj) {
}
/**
* 这也不是一个泛型方法,这也是一个普通的方法,只不过使用了泛型通配符?
*/
public void showKeyValue2(Generic<?> obj) {
}
/**
* 这个方法是有问题的,编译器会为我们提示错误信息:"UnKnown class 'E' "
* 虽然我们声明了<T>,也表明了这是一个可以处理泛型的类型的泛型方法。
* 但是只声明了泛型类型T,并未声明泛型类型E,因此编译器并不知道该如何处理E这个类型。
*
* public <T> T showKeyName(Generic<E> container){
* ...
* }
*/
}
3.泛型方法的使用
class GenericFruit {
static class Fruit {
@Override
public String toString() {
return "fruit";
}
}
static class Apple extends Fruit {
@Override
public String toString() {
return "apple";
}
}
static class Person {
@Override
public String toString() {
return "Person";
}
}
static class GenerateTest<T> {
public void show_1(T t) {
System.out.println(t.toString());
}
// 这是在泛型类中声明了一个泛型方法,使用泛型E,这种泛型E可以为任意类型。可以类型与T相同,也可以不同。
// 由于泛型方法在声明的时候会声明泛型<E>,因此即使在泛型类中并未声明泛型,编译器也能够正确识别泛型方法中识别的泛型<E>。
public <E> void show_3(E t) {
System.out.println(t.toString());
}
//在泛型类中声明了一个泛型方法,使用泛型T,注意这个T是一种全新的类型,可以与泛型类中声明的T不是同一种类型。
public <T> void show_2(T t) {
System.out.println(t.toString());
}
}
public static void main(String[] args) {
Apple apple = new Apple();
Person person = new Person();
// 创建一个泛型类,并且指定泛型的实参是Fruit类型
GenerateTest<Fruit> generateTest = new GenerateTest<Fruit>();
// 调用类的成员方法(这个show_1()方法是类的成员方法)
// apple是Fruit的子类,所以这里可以传入Apple类型
// 单单对泛型T而已, 是可以传入子类对象的(多态)
generateTest.show_1(apple);
// 编译器会报错,因为泛型类型实参在创建的时候指定的是Fruit,而传入的实参类是Person
// generateTest.show_1(person);
// 调用泛型方法
generateTest.show_2(apple);
// 这是完整的写法, 在调用泛型方法的时候指定泛型的类型为Apple
generateTest.<Apple>show_2(apple);
// 调用泛型方法
generateTest.show_2(person);
// 这是完整的写法, 在调用泛型方法的时候指定泛型的类型为Person
generateTest.<Person>show_2(person);
// 这两个也是在调用泛型方法, 与上面的一致
generateTest.show_3(apple);
generateTest.show_3(person);
// 这样调用泛型方法是可以的。指定泛型方法的泛型实参为Fruit类型,然后参入传入Fruit的子类。(多态)
generateTest.<Fruit>show_3(apple);
// 总结:
// a.在指定类的T(也就是泛型实参)的时候, 右侧的Fruit必须和左侧指定的Fruit完全相同,不能写子类。因为:GenericClass<Apple> 与 GenericClass<Fruit>没有继承关系的(具体请看 《 第六节.继承关系 》)
// 比如 GenericClass<Fruit> n = new GenericClass<Fruit>(new Fruit()); 右侧的Fruit必须和左侧指定的Fruit完全相同
// 如果 GenericClass<Fruit> n = new GenericClass<Apple>(new Fruit()); 就报错
// b.在指定类的T(也就是泛型实参)的时候, 传入的方法的参数也一定要是泛型实参的类型,不能是子类。
// 比如 GenericClass<Fruit> n = new GenericClass<>(new Fruit());
// 如果 GenericClass<Fruit> n = new GenericClass<>(new Apple()); 是报错的
// c.但是在调用的时候, 是可以传入 T 的 子类的。
// 比如
// OK
GenericClass<Number> g1 = new GenericClass<Number>(1);
// NOT OK
GenericClass<Number> g2 = new GenericClass<Integer>(2);
// 也 NOT OK
GenericClass<Fruit> gApple = new GenericClass<Fruit>(new Apple());
// 但是在调用方法的时候, 如果参数是类型是T, 那可以传入T的子类的
gApple.setData(new Apple());
}
}
五.泛型使用过程中的相关限制
- 泛型的类型参数只能是类类型(包括自定义类),不能是基本类型
<br /> - 泛型类中的泛型T不能直接实例化, 比如
T t = new T()
, 这样是不行的。
<br /> - 泛型变量不能作为类的静态变量存在, 只能作为类的成员变量。 比如:
private static T t;
这样是不行的。
3.1. 因此,静态方法是无法访问用泛型声明的成员变量的。
3.2. 那么,想在静态方法中引入泛型,就必须使用静态的泛型方法, 也就是staitc <T> void show(T t) {}
这样的形式。
<br /> - 无法使用instanceof关键字判断泛型类的类型, 无法使用“==”判断两个泛型类的实例
List<String> stringArrayList = new ArrayList<String>();
List<Integer> integerArrayList = new ArrayList<Integer>();
/**
* 无法使用instanceof关键字判断泛型类的类型
* Illegal generic type for instanceof
*
* if(stringArrayList instanceof ArrayList<String>){
* return;
* }
*/
/**
* 无法使用“==”判断两个泛型类的实例
* Operator '==' cannot be applied to this two instance
*
* if (stringArrayList == integerArrayList) {
* return;
* }
*/
- 泛型类的原生类型与所传递的泛型无关,无论传递什么类型,原生类是一样的。Java中的泛型,只在编译阶段有效。在编译过程中,正确检验泛型结果后,会将泛型的相关信息擦出,并且在对象进入和离开方法的边界处添加类型检查和类型转换的方法。也就是说,泛型信息不会进入到运行时阶段。
List<String> stringArrayList = new ArrayList<String>();
List<Integer> integerArrayList = new ArrayList<Integer>();
List<Number> numberArrayList = new ArrayList<Number>();
Class classStringArrayList = stringArrayList.getClass();
Class classIntegerArrayList = integerArrayList.getClass();
Class classNumberArrayList = numberArrayList.getClass();
if(classStringArrayList == classIntegerArrayList){
Log.d("泛型测试","类型相同");
}
if(classNumberArrayList == classIntegerArrayList){
Log.d("泛型测试","类型相同");
}
六.继承关系
-
Integer
是Number
的子类, 但是ArrayList<Integer>
不能被看作为ArrayList<Number>
的子类。二者没有继承关系。由此可以看出:同一种泛型可以对应多个版本(因为参数类型是不确定的),不同版本的泛型类实例是不兼容的。因此,需要传入
ArrayList<Number>
作为参数的地方, 不可以传入ArrayList<Integer>
-
泛型类可以继承其它泛型类,例如: public class ArrayList<E> extends AbstractList<E>。
因此,需要传入
List<E>
的地方, 可以传入ArrayList<E>
,E
必须是相同的
示例:
public class GenericInherit<T> {
private T data1;
private T data2;
public T getData1() {
return data1;
}
public void setData1(T data1) {
this.data1 = data1;
}
public T getData2() {
return data2;
}
public void setData2(T data2) {
this.data2 = data2;
}
public static <V> void setData2(GenericInherit<Father> data2) {
}
public static void main(String[] args) {
//Son 继承自 Father
Father father = new Father();
Son son = new Son();
GenericInherit<Father> fatherGenericInherit = new GenericInherit<>();
GenericInherit<Son> sonGenericInherit = new GenericInherit<>();
SubGenericInherit<Father> fatherSubGenericInherit = new SubGenericInherit<>();
SubGenericInherit<Son> sonSubGenericInherit = new SubGenericInherit<>();
/**
* 对于传递的泛型类型是继承关系的泛型类之间是没有继承关系的
* GenericInherit<Father> 与GenericInherit<Son> 没有继承关系
* Incompatible types.
*/
father = new Son();
//fatherGenericInherit = new GenericInherit<Son>();
/**
* 泛型类可以继承其它泛型类,例如: public class ArrayList<E> extends AbstractList<E>
*/
fatherGenericInherit = new SubGenericInherit<Father>();
/**
* 泛型类的继承关系在使用中同样会受到泛型类型的影响
* setData2()方法的参数是: GenericInherit<Father>, 可以传入它的子类 SubGenericInherit<Father>
* 但是 GenericInherit<Son> 、SubGenericInherit<Son> 与 GenericInherit<Father> 都没有任何继承关系的
*/
setData2(fatherGenericInherit);
//setData2(sonGenericInherit);
setData2(fatherSubGenericInherit);
//setData2(sonSubGenericInherit);
}
private static class SubGenericInherit<T> extends GenericInherit<T> {
}
八.泛型的上下边界和通配符 ?
public class GenericByWildcard {
private static void print(GenericClass<Fruit> fruitGenericClass) {
System.out.println(fruitGenericClass.getData().getColor());
}
private static void use() {
GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
print(fruitGenericClass);
GenericClass<Orange> orangeGenericClass = new GenericClass<>();
//类型不匹配,可以使用<? extends Parent> 来解决
//print(orangeGenericClass);因为GenericClass<Orange> 和 GenericClass<Fruit>没有继承关系
}
/**
* <? extends Parent> 指定了泛型类型的上届
*/
private static void printExtends(GenericClass<? extends Fruit> genericClass) {
System.out.println(genericClass.getData().getColor());
}
public static void useExtend() {
GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
printExtends(fruitGenericClass);
GenericClass<Orange> orangeGenericClass = new GenericClass<>();
printExtends(orangeGenericClass);
GenericClass<Food> foodGenericClass = new GenericClass<>();
//Food是Fruit的父类,超过了泛型上届范围,类型不匹配
//printExtends(foodGenericClass);
//表示GenericClass的类型参数的上届是Fruit
GenericClass<? extends Fruit> extendFruitGenericClass = new GenericClass<>();
Apple apple = new Apple();
Fruit fruit = new Fruit();
/*
* 道理很简单,? extends X 表示类型的上界,类型参数是X的子类,那么可以肯定的说,
* get方法返回的一定是个X(不管是X或者X的子类)编译器是可以确定知道的。
* 但是set方法只知道传入的是个X,至于具体是X的那个子类,不知道。
* 总结:主要用于安全地访问数据,可以访问X及其子类型,并且不能写入非null的数据。
*/
//extendFruitGenericClass.setData(apple);
//extendFruitGenericClass.setData(fruit);
fruit = extendFruitGenericClass.getData();
}
/**
* <? super Child> 指定了泛型类型的下届
*/
public static void printSuper(GenericClass<? super Apple> genericClass) {
System.out.println(genericClass.getData());
}
public static void useSuper() {
GenericClass<Food> foodGenericClass = new GenericClass<>();
printSuper(foodGenericClass);
GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
printSuper(fruitGenericClass);
GenericClass<Apple> appleGenericClass = new GenericClass<>();
printSuper(appleGenericClass);
GenericClass<HongFuShiApple> hongFuShiAppleGenericClass = new GenericClass<>();
// HongFuShiApple 是Apple的子类,达不到泛型下届,类型不匹配
//printSuper(hongFuShiAppleGenericClass);
GenericClass<Orange> orangeGenericClass = new GenericClass<>();
// Orange和Apple是兄弟关系,没有继承关系,类型不匹配
//printSuper(orangeGenericClass);
//表示GenericClass的类型参数的下界是Apple
GenericClass<? super Apple> supperAppleGenericClass = new GenericClass<>();
supperAppleGenericClass.setData(new Apple());
supperAppleGenericClass.setData(new HongFuShiApple());
/*
* ? super X 表示类型的下界,类型参数是X的超类(包括X本身),
* 那么可以肯定的说,get方法返回的一定是个X的超类,那么到底是哪个超类?不知道,
* 但是可以肯定的说,Object一定是它的超类,所以get方法返回Object。
* 编译器是可以确定知道的。对于set方法来说,编译器不知道它需要的确切类型,但是X和X的子类可以安全的转型为X。
* 总结:主要用于安全地写入数据,可以写入X及其子类型。
*/
//supperAppleGenericClass.setData(new Fruit());
//get方法只会返回一个Object类型的值。
Object data = supperAppleGenericClass.getData();
}
/**
* <?> 指定了没有限定的通配符
*/
public static void printNonLimit(GenericClass<?> genericClass) {
System.out.println(genericClass.getData());
}
public static void useNonLimit() {
GenericClass<Food> foodGenericClass = new GenericClass<>();
printNonLimit(foodGenericClass);
GenericClass<Fruit> fruitGenericClass = new GenericClass<>();
printNonLimit(fruitGenericClass);
GenericClass<Apple> appleGenericClass = new GenericClass<>();
printNonLimit(appleGenericClass);
GenericClass<?> genericClass = new GenericClass<>();
//setData 方法不能被调用, 甚至不能用 Object 调用;
//genericClass.setData(foodGenericClass);
//genericClass.setData(new Object());
//返回值只能赋给 Object
Object object = genericClass.getData();
}
}
PECS法则
PECS指“Producer Extends,Consumer Super”。换句话说,如果参数化类型表示一个生产者,就使用<? extends T>;如果它表示一个消费者,就使用<? super T>。
- 如果你想从一个数据类型里获取数据,那么它就是生产者,我们要从里面get()数据,这样的使用 ? extends 通配符2. 如果你想把对象写入一个数据结构里,那么他就是消费者,我们要把数据丢入这个数据结构(set()), 这时候使用 ? super 通配符
- 如果你既想存,又想取,那就别用通配符。
PESC法则请参考:
九.获取泛型的参数类型
参考: TypeToken
十.Gson中的TypeToken
参考: TypeToken
总结的内容,大部分参考并摘录了:
https://www.jianshu.com/p/986f732ed2f1
https://blog.csdn.net/s10461/article/details/53941091