因为看书的时候对这部分知识没太看懂。网上查了一下才明白。书上讲的有点乱。还是喜欢c++ primer这种书= =感觉写的比较清楚。
8.5 泛型代码
类型擦除:虚拟机没有泛型对象,在定义一个泛型类型时,会自动提供其相对应的原始类型,即擦去类型变量,如果是限定的类型变量,则用第一个限定类型的代替,如果是无限定的类型变量,则用Object代替。
//Pair<T>的原始类型:
public class pair
{
private Object first;
private Object second;
public pair()
{first = null;second = null;}
public pair(Object first,Object second)
{this.first = first;this.second = second;}
public Object getFirst()
{return first;}
public Object getSecond()
{return second;}
public void setFirst(Object newinput)
{first = newinput;}
public void setSecond(Object newinput)
{second = newinput;}
}
//限定的泛型类的原始类型
public class Interval<T extends Comparable & Serializable> implements Serializable
{
private T lower;
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
- 9
- 10
- 11
- 12
- 13
- 14
- 15
- 16
- 17
- 18
- 19
- 20
- 21
- 22
- 23
- 24
在调用泛型方法时,如果擦去了返回类型,则编译器会插入强制类型转换
Pair<Employee> buddies = ...;
Employee = buddies.getFirst();//getFirst()首先生成一个object类,再强制转换为Employee类
- 1
- 2
桥方法
泛型方法中也会发生类型消除
如
class DateInterval extends Pair<LocalDate>
{
public void setSecond(LocalDate second)
{
if(second.compareTo(getFirst()) >= 0)
super.setSecond(second);
}
}
- 1
- 2
- 3
- 4
- 5
- 6
- 7
- 8
上面此段代码有个问题,DateInterval是Pair< LocalDate>的子类,其中有一个方法setSecond(),通常来说子类的方法需覆盖住超类的方法,然而由于类型消除,子类会从超类的原始类型中继承到另一个setSecond()
public void setSecond(LocalDate second)//child-class
public void setSecond(Object second)//继承
//当调用setSecond(aDate)时,多态性会出现问题
- 1
- 2
- 3
此时子类的方法不能覆盖住超类的方法,从而影响了多态性。解决方法便是编译器在子类中生成了一个桥方法,从而调用了子类中的setSecond
public void setSecond(Object second)
{setSecond((LocalDate) second);}
- 1
- 2
换一种情况来看
class DateInterval extends Pair<LocalDate>
{
public LocalDate getSecond()
{return (Date) super.getSecond().clone();}
}
- 1
- 2
- 3
- 4
- 5
此时编译器也会生成一个桥方法
public Object getSecond()
{...}
- 1
- 2
子类中便有两个不含参数且同名的方法了
这在编写上是不允许的,因为它们没有参数,只是返回类型不同,而在编译器中,是以参数类型和返回类型来确定一个方法,因此编译器可以参数两个仅返回类型不同的方法字节码,但是这样的编写是不能通过编译的。
8.6 约束
1.不能用基本类型实例化类型参数,即不能有Pair< double>,因为类型擦去后Pair的类型为Object,不能将double赋给它,只能有Pair< Double>
2.对于类型查询(instanceof),只产生原始类型。getClass()也是返回原始类型
if(a instanceof Pair<String>){...}//实际上只能查询a是否属于Pair
- 1
3.不能创建参数化类型的数组,即new Pair< String>[10]是不允许的,Pair< String>[] 不能初始化
只能使用ArrayList< Pair< String>>来搜集参数化类型
4.由于不能创建参数化类型的数组,所以在提供一系列参数化类型作为参数的方法中,调用用时会发出警告,可用@SafeVarags来消除创建泛型数组的有关限制
@SafeVarags
public static <T> void addAll(collection<T> coll,T...st)//一系列T对象会被写为数组
- 1
- 2
5.不能实例化类型变量,即new T(..)/new T[…]/T.class等
6.不能构造泛型数组,一般需要强制类型转换,即[T[]] new Object[size];
7.不能在静态域和静态方法中使用类型变量
8.7 继承
如果S为T的超类,Pair< S>和Pair< T>没有任何关系
8.8 通配符类型
通配符可以分为子类型限定通配符和超类型限定通配符
子类型限定通配符: ? extends Employee 不能向更改器方法传递特定的类型,但是可以让访问器方法返回特定类型
超类型限定通配符:? super Manager,不能像向改器传递Manager的超类,访问器方法只能返回Object类