泛型类型的子类及通配符的使用

本文讲述以下几个方面的内容,试图说明泛型类型的子类及通配符的使用。

  (1)    子类及替换原则;

  (2)    使用extends关键字的通配符;

  (3)    使用super关键字的通配符;

  (1)    子类及替换原则

  在java语言中,我们通俗讲一个类是另一个类的子类型,是通过使用extends关键字去继承某一个类或者使用implements关键字去实现某些接口。这样我们在编程时就可以面向接口或基类进行编程,如:

  Number num1 = new Integer(1);

  Number num2 = new Double(2.1d);

  这个就是所谓的替换原则,替换原则的定义是:

  Substitution Principle: a variable of a given type may be assigned a value of any subtype of that type, and a method with a parameter of a given type may be invoked with an argument of any subtype of that type.

  大概的意思是说某种类型的变量可以被该类型的任何子类所赋值,一个方法中的参数也可以被该参数的任何子类进行调用。

  现在我们再来看看泛型中替换原则的适用性:

  Example2.1

  List<Number> listNums = new ArrayList<Number>();

  nums.add(10);

  nums.add(8.88);

  在example2.1 中,可以看出替换原则被很好地应用在这里,ArrayList是List的子类,我们提供给listNums变量的类型参数为Number,往 listNums中添加元素时,10被封箱为Integer类型,而Integer是Number的子类,第三行的情况类似。

  Example2.2

  List<Integer> intList = new ArrayList<Integer>();

  List<Number> numList = intList;   //compile error

  …

  numList.add(1.35); // can‘t do that

  根据替换原则,我们会很容易想到,既然Integer是Number的子类,则我们应该可以将List<Integer>的变量赋给 List<Number>的变量,但从实际情况表明,List<Integer>并不是List<Number>的子类。

  我们不妨试想想,若果List<Integer>类型的变量可以成功赋值给List<Number>类型的变量,会出现什么情况?我们可能在程序的某个位置添加一个double类型的元素进去numList中,而实质上在numList中其它元素都是Integer 的类型的元素,这样就违背了泛型的初衷了。

  有时,我们确实希望将形如List<Integer>的List对象赋给List<Number>的变量,这时就要使用extends关键字的通配符。

  (2)    使用extends关键字的通配符

  Example2.3

  List<Integer> intList = new ArrayList<Integer>();

  List<? extends Number> numList = intList();

  …

  numList.add(1.35); //compile error (can‘t do that)

  从Example2.3看到numList这个变量,我们可以将类型参数为Number及其Number子类的List赋给它。

  记住一条规则如果你使用了“? extends T”,一般情况下,你不能往该数据结构中put元素,而你可以做的就是get元素。

  如果要往内put元素,就需要使用下面提到的super关键字的通配符。

  (3)    使用super关键字的通配符

  Example2.4

  List<Integer> intList = new ArrayList<Integer();

  List<? super Integer> numList = intList;

  numList.add(3); //can put integer or null

  在example2.4 我们可以看到<? super Integer>的意思为,我们可以将类型参数为Integer或Integer超类的List赋给 numList变量,并且可以put元素到列表中(注意:在该例子中put进的元素只能为Integer或null类型)。

  一条比较通用的规则:如果要往List中put元素则用<? super T>,如果要从List中get元素则用<? extends T>,如果既要get又要put则不使用通配符。

 
  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
泛型的产生条件 泛型是为了解决在编译期间无法确定类型而引入的,其产生条件包括: 1. 在代码需要使用泛型类,但是这个泛型类的具体类型在编译期间是不确定的; 2. 在代码需要对泛型类进行操作,比如调用其方法或者获取其属性值。 泛型的概念 泛型是指在编写代码时,不需要指定数据类型,而是在使用时再指定具体的数据类型。这样就可以提高代码的复用性和灵活性。 泛型类 泛型类是指使用泛型定义的类,其泛型参数可以用在类的成员变量、成员方法、构造方法泛型类派生类 泛型类派生类是指使用泛型定义的类的子类,其子类可以继续使用父类定义的泛型类。 带泛型子类泛型子类是指在继承泛型类时,子类也要使用泛型类。 不带泛型子类 不带泛型子类是指在继承泛型类时,子类使用泛型类泛型接口 泛型接口是指使用泛型定义的接口,其泛型参数可以用在接口的方法泛型方法 泛型方法是指使用泛型定义的方法,其泛型参数可以用在方法的参数列表、返回值、方法体类型通配符 类型通配符是指在定义泛型使用的一种特殊符号,用于表示不确定的类型。 引出类型通配符 类型通配符可以用于引出泛型类参数的上限或者下限。 类型通配符的上限 类型通配符的上限是指使用 extends 关键字限制泛型类参数的范围,表示泛型类参数必须是某个类型子类或者实现类。 类型通配符的下限 类型通配符的下限是指使用 super 关键字限制泛型类参数的范围,表示泛型类参数必须是某个类型的父类或者超类。 类型擦除 类型擦除是指在编译期间,将泛型类参数替换为其上限或者 Object 类型的过程。 无限制类型的擦除 无限制类型的擦除是指在泛型类参数没有明确指定上限或者下限时,将其擦除为 Object 类型。 有限制类型擦除 有限制类型擦除是指在泛型类参数有明确指定上限或者下限时,将其擦除为上限或者下限。 擦除泛型方法类型定义的参数 在泛型方法,如果定义了泛型类参数,则在编译期间也会进行类型擦除。 桥接方法 在泛型类或者泛型接口,如果有泛型方法,则在编译期间会自动生成桥接方法来确保类型安全。 泛型数组 泛型数组是指使用泛型定义的数组,其数组元素的类型泛型类参数。 泛型与反射 泛型与反射的结合可以实现动态创建泛型类对象、获取泛型类信息等功能。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值