泛型类指可以接受类型参数的类。泛型类在集合类中被广泛使用。

定义一个泛型类

泛型类使用方括号 [] 来接受类型参数。一个惯例是使用字母 A 作为参数标识符,当然你可以使用任何参数名称。

  1. class Stack[A] {
  2. private var elements: List[A] = Nil
  3. def push(x: A) { elements = x :: elements }
  4. def peek: A = elements.head
  5. def pop(): A = {
  6. val currentTop = peek
  7. elements = elements.tail
  8. currentTop
  9. }
  10. }

上面的 Stack 类的实现中接受类型参数 A。 这表示其内部的列表,var elements: List[A] = Nil,只能够存储类型 A 的元素。方法 def push 只接受类型 A 的实例对象作为参数(注意:elements = x :: elementselements 放到了一个将元素 x 添加到 elements 的头部而生成的新列表中)。

使用

要使用一个泛型类,将一个具体类型放到方括号中来代替 A

  1. val stack = new Stack[Int]
  2. stack.push(1)
  3. stack.push(2)
  4. println(stack.pop) // prints 2
  5. println(stack.pop) // prints 1

实例对象 stack 只能接受整型值。然而,如果类型参数有子类型,子类型可以被传入:

  1. class Fruit
  2. class Apple extends Fruit
  3. class Banana extends Fruit
  4. val stack = new Stack[Fruit]
  5. val apple = new Apple
  6. val banana = new Banana
  7. stack.push(apple)
  8. stack.push(banana)

Apple 和类 Banana 都继承自类 Fruit,所以我们可以把实例对象 applebanana 压入栈 Fruit 中。

注意:泛型类型的子类型是不可传导的。这表示如果我们有一个字母类型的栈 Stack[Char],那它不能被用作一个整型的栈 Stack[Int]。否则就是不安全的,因为它将使我们能够在字母型的栈中插入真正的整型值。结论就是,只有当类型 B = A 时, Stack[A]Stack[B] 的子类型才成立。因为此处可能会有很大的限制,Scala 提供了一种 类型参数注释机制 用以控制泛型类型的子类型的行为。