#Java 核心技术卷一阅读笔记# 第八章 泛型程序设计

18 篇文章 0 订阅

8.2 泛型类

泛型:具有类型参数(类型变量),可以用其他具体的类来替代此参数,从而扩大方法的实用范围以及提高程序的安全性。
泛型类:具有一个或多个类型变量的类,用具体的类替换类型变量即为实例化泛型类。

public class pair<T>
        {
            private T first;
            private T second;
            public pair()
            {first = null;second = null;}
            public pair(T first,T second)
            {this.first = first;this.second = second;}
            public T getFirst()
            {return first;}
            public T getSecond()
            {return second;}
            public void setFirst(T newinput)
            {first = newinput;}
            public void setSecond(T newinput)
            {second = newinput;}
        }
public class Pair<T,U>//两个以上类型变量用,隔开
{
...
}

8.3 泛型方法

参数或返回类型为泛型变量的方法,可以位于普通类或泛型类之间,< T>位于修饰符之后、返回类型之前。

class Alg
{
    public static <T> T getMiddle(T...a){
        return a[a.length/2];
    }
}

调用此方法时,可以省略具体类型,编译器可以根据上下内容推断出来

String res = Alg.<String> getMiddle("1","2","3");//标明实例化的具体类型
String res = Alg.getMiddle("1","2","3");//如果参数类型不同,则会寻找共同的超类型
double m = Alg.getMiddle(1,2.5,5);//得出错误信息:cannot convert from Number&Comparable<?> to double,其共同超类型为Number/Comparable

8.4 类型变量的限制

< T extends BoundingType>
将泛型限制为实现了指定接口或类的类,限定类型可以是接口,也可以是类。能够限定多个接口,但最多只能限定一个类,且该类必须是限定列表中的第一个。多个限定类型之间用&隔开。

public static <T extends Comparable> T min(T[] a)//只有实现了Comparable接口的类才能调用此泛型方法,从而保证每个调用方法的类都有compareTo()方法

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;

}

在调用泛型方法时,如果擦去了返回类型,则编译器会插入强制类型转换

Pair<Employee> buddies = ...;
Employee = buddies.getFirst();//getFirst()首先生成一个object类,再强制转换为Employee类

桥方法
泛型方法中也会发生类型消除

class DateInterval extends Pair<LocalDate>
{
    public void setSecond(LocalDate second)
    {
        if(second.compareTo(getFirst()) >= 0)
            super.setSecond(second);
    }
}

上面此段代码有个问题,DateInterval是Pair< LocalDate>的子类,其中有一个方法setSecond(),通常来说子类的方法需覆盖住超类的方法,然而由于类型消除,子类会从超类的原始类型中继承到另一个setSecond()

public void setSecond(LocalDate second)//child-class
public void setSecond(Object second)//继承
//当调用setSecond(aDate)时,多态性会出现问题

此时子类的方法不能覆盖住超类的方法,从而影响了多态性。解决方法便是编译器在子类中生成了一个桥方法,从而调用了子类中的setSecond

public void setSecond(Object second)
{setSecond((LocalDate) second);}

换一种情况来看

class DateInterval extends Pair<LocalDate>
{
    public LocalDate getSecond()
    {return (Date) super.getSecond().clone();}
}

此时编译器也会生成一个桥方法

public Object getSecond()
{...}

子类中便有两个不含参数且同名的方法了
这在编写上是不允许的,因为它们没有参数,只是返回类型不同,而在编译器中,是以参数类型和返回类型来确定一个方法,因此编译器可以参数两个仅返回类型不同的方法字节码,但是这样的编写是不能通过编译的。

8.6 约束

1.不能用基本类型实例化类型参数,即不能有Pair< double>,因为类型擦去后Pair的类型为Object,不能将double赋给它,只能有Pair< Double>
2.对于类型查询(instanceof),只产生原始类型。getClass()也是返回原始类型

if(a instanceof Pair<String>){...}//实际上只能查询a是否属于Pair

3.不能创建参数化类型的数组,即new Pair< String>[10]是不允许的,Pair< String>[] 不能初始化
只能使用ArrayList< Pair< String>>来搜集参数化类型
4.由于不能创建参数化类型的数组,所以在提供一系列参数化类型作为参数的方法中,调用用时会发出警告,可用@SafeVarags来消除创建泛型数组的有关限制

@SafeVarags
public static <T> void addAll(collection<T> coll,T...st)//一系列T对象会被写为数组

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类

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值