第五章泛型

目录

  • 不要使用原始类型
  • 消除非检查警告
  • 列表优于数组
  • 优先考虑泛型
  • 优先使用泛型方法
  • 使用限定通配符来增加API的灵活性
  • 合理地结合泛型和可变参数
  • 优先考虑类型安全的异构容器

泛型

不要使用原始类型

  • 如果你使用原始类型,则会丧失泛型的所有安全性和表达上的优势
  • 你必须在类字面值(class literals)中使用原始类型。egList.class,String [] .class和int.class都是合法的。o instanceof Set需要使用原始类型。
  • 限制类型参数<E extends Number》,递归类型限制<T extends Comparable<T>>,限制通配符类型List<? extends Number>,泛型方法 static <E> List<E> asList(E[] a)

消除非检查警告

  • 如果你不能消除警告,但你可以证明引发警告的代码是类型安全的,那么(并且只能这样)用@SuppressWarnings(“unchecked”)注解来抑制警告
  • 声明一个局部变量来保存返回值并标注它的声明
@SuppressWarnings("unchecked") T[] result =
            (T[]) Arrays.copyOf(elements, size, a.getClass());
retur result;

列表优于数组

  • 数组提供运行时类型的安全性,但不提供编译时类型的安全性,反之亦然。 一般来说,数组和泛型不能很好地混合工作。 如果你发现把它们混合在一起,得到编译时错误或者警告,你的第一个冲动应该是用列表来替换数组
  • 数组是协变的,泛型是收约束的。
// 运行时报错
Object[] objectArray = new Long[1];
objectArray[0] = "I don't fit in"; // Throws ArrayStoreException

// 无法编译通过
List<Object> ol = new ArrayList<Long>(); // Incompatible types
ol.add("I don't fit in");
  • 数组提供了运行时类型安全性,不保证编译时安全性,泛型则反过来

优先考虑泛型

  • 尝试创建Stack<int>或Stack<double>将导致编译时错误。 这是Java泛型类型系统的一个基本限制
  • 泛型类型比需要在客户端代码中强制转换的类型更安全,更易于使用。 当你设计新的类型时,确保它们可以在没有这种强制转换的情况下使用
  • 这个类应该已经被参数化了,但是由于事实并非如此,我们可以对它进行泛型化。 就目前而言,客户端必须强制转换从堆栈中弹出的对象,而这些强制转换可能会在运行时失败。
public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 16;

    public Stack() {
        elements = new Object[DEFAULT_INITIAL_CAPACITY];
    }

    public void push(Object e) {
        ensureCapacity();
        elements[size++] = e;
    }

    public Object pop() {
        if (size == 0)
            throw new EmptyStackException();
        Object result = elements[--size];
        elements[size] = null; // Eliminate obsolete reference
        return result;
    }
  • 可参考Stack对这种泛型的写法,它解决了初始化不能用new E()的问题,用到了在返回ele时强转Object到E。

优先使用泛型方法

  • 泛型方法比需要客户端对输入参数和返回值进行显式强制转换的方法更安全,更易于使用。 像类型一样,你应该确保你的方法可以不用强制转换,这通常意味着它们是泛型的。
// 这种会有警告
public static Set union(Set s1, Set s2) {

    Set result = new HashSet(s1);

    result.addAll(s2);

    return result;
}

//这种是安全的
public static <E> Set<E> union(Set<E> s1, Set<E> s2) {

    Set<E> result = new HashSet<>(s1);

    result.addAll(s2);

    return result;

}
  • 递归类型限制
public static <E extends Comparable<E>> E max(Collection<E> c) {
    if (c.isEmpty()) throw new IllegalArgumentException("Empty collection");
    E result = null;
    for (E e : c){
        if (result == null || e.compareTo(result) > 0){
            result = Objects.requireNonNull(e);
        }
    }
    return result;
}

使用限定通配符来增加API的灵活性

  • 参数化类型是不变的。换句话说,对于任何两个不同类型的Type1和Type,List <Type1>既不是List <Type2>子类型也不是其父类型
  • 为了获得最大的灵活性,对代表生产者或消费者的输入参数使用通配符类型
//考虑以下这种,如果pushAll方法是public void pushAll(Iterable<E> src) 则会报错,因为参数化类型是不变的。
Stack<Number> numberStack = new Stack<>();

Iterable<Integer> integers = ... ;

numberStack.pushAll(integers);

//可以变成这种, 这样增加灵活性。生成栈使用的E实例使用extend
public void pushAll(Iterable<? extends E> src)
// 消费栈使用的E实例使用super 
public void popAll(Collection<? super E> dst)
// 所有Comparable和Comparator都是消费者
  • 返回类型仍然是Set <E>。 不要使用限定通配符类型作为返回类型
  • 灵活性体现
// 无界类型参数
public static <E> void swap(List<E> list, int i, int j); 

// 无界通配符:该方式优于上一个方式,但是由于无界通配符类型无法修改,即需要借助helper进行修改,但这对于调用者无需关心。
public static void swap(List<?> list, int i, int j) { 
    swapHelper(list, i, j);
}

private static <E> void swapHelper(List<E> list, int i, int j) { 
    list.set(i, list.set(j, list.get(i)));
}
汇总
  1. 上边界类型通配符(<? extends 父类型>):因为可以确定父类型,所以可以以父类型去获取数据(向上转型)。但是不能写入数据。
  2. 下边界类型通配符(<? super 子类型>):因为可以确定最小类型,所以可以以最小类型去写入数据(向上转型)。而不能获取数据。
  3. 无边界类型通配符(<?>) 等同于 上边界通配符<? extends Object>,所以可以以Object类去获取数据。List list 相当于List<Object> list

合理地结合泛型和可变参数

  • 可变参数和泛型不能很好地交互,因为可变参数机制是在数组(协变,)上面构建的脆弱的抽象,并且数组具有与泛型不同的类型规则。 虽然泛型可变参数不是类型安全的,但它们是合法的。 如果选择使用泛型(或参数化)可变参数编写方法,请首先确保该方法是类型安全的

优先考虑类型安全的异构容器

  • 泛型API的通常用法(以集合API为例)限制了每个容器的固定数量的类型参数。 你可以通过将类型参数放在键上而不是容器上来解决此限制。 可以使用Class对象作为此类型安全异构容器的键。 以这种方式使用的Class对象称为类型令牌。
  • 与普通Map不同,所有的键都是不同的类型。 因此,我们将Favorites称为类型安全异构容器
public class Favorites {
    private Map<Class<?>, Object> favorites = new HashMap<>();

    public <T> void putFavorite(Class<T> type, T instance) {
        favorites.put(Objects.requireNonNull(type), instance);
    }

    public <T> T getFavorite(Class<T> type) {
        return type.cast(favorites.get(type));
    }
}
  • 如果你尝试保存你最喜欢的List <String>,程序将不能编译。 原因是无法获取List <String>的Class对象。 List <String> .class是语法错误,也是一件好事。 List <String>和List <Integer>共享一个Class对象
  • 现有实践是把各个Service类作为key

参考文章

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 泛型的作用:告诉编译器每个集合中可接受哪些对象类型,编译器自动地为你的插入进行转化,并在编译时告知是否插入错误的对...
    Timorous阅读 253评论 0 0
  • Java1.5中添加了泛型。没有泛型之前,从集合中读取的每一个对象都必须进行转换,如果不小心添加了类型错误的对象,...
    bunnypu阅读 495评论 0 0
  • 泛型是.NET Framework2.0新增的一个特性,在命名空间System.Collections.Gener...
    张中华阅读 313评论 0 3
  • 第8章 泛型 通常情况的类和函数,我们只需要使用具体的类型即可:要么是基本类型,要么是自定义的类。但是在集合类的场...
    光剑书架上的书阅读 2,158评论 6 10
  • object 变量可指向任何类的实例,这让你能够创建可对任何数据类型进程处理的类。然而,这种方法存在几个严重的问题...
    CarlDonitz阅读 932评论 0 5