读书笔记:Effective Java-第5章 泛型Generics

11 篇文章 0 订阅
11 篇文章 0 订阅

目录

Item 26: Don't use raw types

Item 27: Eliminate unchecked warnings

Item 28: Prefer lists to arrasy

Item 29: Favor generic types

Item 30: Favor generic methods

Item 31: Use bounded wildcards to increase API flexibility

Item 33: Consider typesafe heterogeneous containers

参考


Item 26: Don't use raw types

不要使用原生态类型

泛型包含泛型类和泛型接口。

原生态类型指List这种没有指定具体类型参数的泛型,如List<E>的raw type是List、Collection<E>的是Collection、Iterator<E>的是Iterator等等。

原生类型raw type指不带任何实际类型参数的泛型名称,它们的存在是为了兼容泛型出现之前的代码。

如果使用原生态类型,会失掉泛型在安全性和描述性方面的优势。

不能将任何元素(除null外)放到无限制通配类型(如Collections<?>)中,如下:

// 错误代码
void test(Set<?> s) {
    s.add("e");  // 会报错
}

必须在类字面量(class literal)中使用原生态类型。List.class、String[].class、int.class合法,但List<String>.class、List<?>.class不合法。

泛型相关术语表

术语示例参见条目
参数化的类型 parameterized typeList<String>Item 26
实际类型参数 actual type parameterStringItem 26
泛型 generic typeList<E>Item 26, 29
形式类型参数 formal type parameterEItem 26
无限制通配符类型 unbounded wildcard typeList<?>Item 26
原生态类型 raw typeListItem 26
有限制类型参数 bounded type parameter<E extends Number>Item 29
递归类型限制 recursive type bound<T extends Comparable<T>>Item 30
有限制通配符类型 Bounded wildcard typeList<? extends Number>Item 31
泛型方法 Generic methodstatic <E> List<E> asList(E[] a)Item 30
类型令牌 Type tokenString.classItem 33

Item 27: Eliminate unchecked warnings

消除非受检的警告

要尽可能消除每一个非受检警告。如果无法消除,但可以证明是类型安全的,可以用注解@SuppressWarnings("unchecked")来禁止这类警告,同时添加注释说明原因。

应该在尽可能小的范围使用@SuppressWarnings。

Item 28: Prefer lists to arrasy

列表优先于数组

数组array是协变的(covariant):如果Sub为Super的子类,那么数组类型Sub[]也是Super[]的子类。

泛型是不可变的(invariant):任意两个不同类型Type1、Type2,即使它们存在继承关系,List<Type1>和List<Tpye2>都不存在超-子(父-子)类型关系。

数组是可以具体化的reified,即在编译时和运行时都知道它们的元素类型。

泛型是通过擦除(erasure)来实现的,不可具体化的(non-reifiable),即只有在编译时才知道它们的元素类型,在运行时丢弃(或擦除)元素的类型信息。擦除主要是为了兼容历史代码。

因此,创建泛型数组是非法的,如new List<E>[]、new E[]、new List<String>[]都会出现“generic array creation errors”。

Item 29: Favor generic types

优先考虑泛型

示例:

public class Stack<E> {
    private E[] elements;
    public Stack() {
        elements = new E[16];
    }
    ...
}

Item 30: Favor generic methods

优先考虑泛型方法

示例:

public static <E> Set<E> union(Set<E> s1, Set<E> s2) {
    Set<E> result = new HashSet<>(s1);
    result.addAll(s2);
    return result;
}

Item 31: Use bounded wildcards to increase API flexibility

利用有限制通配符来提升API的灵活性

有限制通配符有2种:<? extends E>和<? super E>。

使用口诀:PECS,即producer-extends、comsumer-super。

示例:

// producer-extends情况
public void pushAll(Collection<? extends E> src) {
    for(E e : src) {
        push(e);  // 泛型参数变量作为生产者,给外界提供数据    
    }
}

// consumer-super情况
public void popAll(Collection<? super E> dst) {
    while(!isEmpty()) {
        dst.add(pop());  // 泛型参数变量作为消费者,消费外界数据    
    }
}

Item 32: Combine generics and varargs judiciously

谨慎并用泛型和可变参数

Item 33: Consider typesafe heterogeneous containers

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

泛型的一般用法有个限制,即每个容器只能有固定数量的类型参数,而异构容器通过将类型参数放到key值而非容器上绕开此限制,如:

// 类型安全的异构容器模式
public class Favorites {
    private Map<Class<?>, Object> favorites = new HashMap<>();
    public <T> void putFavorite(Class<T> type, T instance) {
        favorites.put(Objects.reguireNonNull(type), instance);
    }
    public <T> T getFavorite(Class<T> type) {
        return type.cast(favorites.get(type));
    }
}

// 客户端代码
Favorites f = new Favorites();
f.putFavorite(String.class, "Java");
f.putFavorite(Integer.class, 0xcafebabe);
f.putFavorite(Class.class, Favorites.class);
Class<?> favoriteClass = f.getFavorite(Favorites.class);

参考

Java泛型类型擦除 - Java泛型教程 | 编程字典

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: jmu-java-05集合(泛型)-10-generalstack是关于Java泛型中的通用栈的学习内容。通用栈是一种可以存储任意类型数据的栈结构,通过泛型的方式实现。在学习中,我们可以了解到通用栈的实现原理、使用方法以及注意事项等内容,帮助我们更好地理解和应用Java泛型。 ### 回答2: JMU-Java-05集合(泛型)-10-GeneralStack是一个Stack(栈)的实现类,使用Java中的泛型进行定义,可以存储任何类型的数据。 在该类中,使用一个Object类型的数组进行存储元素,并通过一个整型变量top来表示当前栈顶元素的下标。在push(入栈)方法中,先将top加一,然后将元素存储在数组中,实现了入栈的功能。在pop(出栈)方法中,先判断栈是否为空,若为空则抛出栈空异常,否则将top减一,并返回数组中相应的元素,实现了出栈的功能。其他方法如isEmpty(判断栈是否为空)、isFull(判断栈是否已满)和size(获取栈中元素个数)也在该类中实现。 该类的泛型定义使得我们可以使用该类存储任何类型的数据,而不需要在定义类时指定数据类型,提高了其灵活性和可复用性。在使用时,我们只需要在创建对象时传入相应的数据类型,如:GeneralStack<Integer> stack = new GeneralStack<Integer>();即可。 该类还实现了Iterable接口,使得该类可以使用foreach循环进行遍历操作,方便了我们对栈中元素的操作。同时,该类还通过对数组的动态扩容,解决了数组固定大小的限制问题,从而提高了该栈类的通用性和易用性。 总之,JMU-Java-05集合(泛型)-10-GeneralStack是一个使用泛型实现的通用栈类,可以存储各种类型的数据,并提供了常用的栈操作方法,具有较高的可复用性与适用性。 ### 回答3: jmu-java-05集合(泛型)-10-generalstack,意为基于泛型的栈实现。 首先我们需要了解什么是泛型泛型Java SE 5引入的一个新特性,它可以让我们在编写代码时定义一些未知的类型参数,以达到代码的复用和类型安全的目的。对于集合类或者其他容器类而言,适用于任何类的容器的需求是普遍存在的,这时就可以应用泛型。 在Java泛型使用<>标识,其语法格式如下: ```java public class 类名<类型参数列表> { //成员变量、方法等 } ``` 类型参数列表是由逗号隔开的参数列表,可以理解为未知类型的占位符。这样的好处是可以在编写集合类时,避免出现类型转换等一系列问题,提高程序的可读性。 而在jmu-java-05集合(泛型)-10-generalstack中,则是基于泛型实现的数据结构——栈。栈是一种后进先出(LIFO)的数据结构,它只允许在栈顶添加或删除元素,因此操作非常简单、快速并且高效。 在该实现中,栈的元素可以是任意类型,数据元素入栈则是通过 push() 方法实现的,出栈则是通过 pop() 方法实现的。其中,push() 方法用于向栈中添加新的元素,pop() 方法用于弹出并返回栈顶的元素。同时,还提供了 isEmpty() 方法、size() 方法等基本的栈操作方法。 总之,jmu-java-05集合(泛型)-10-generalstack是一种基于泛型实现的栈,提供了复用和类型安全的目的。其实现使用简便、高效、易于扩展等优点,广泛应用在Java开发中。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值