特征用于在類之間共享接口和字段。類似于Java 8的接口。類和對(duì)象可以擴(kuò)展特征,但是特征不能實(shí)例化,因此也沒(méi)有參數(shù)。
定義特征
最簡(jiǎn)單的定義方法是trait關(guān)鍵字加上一個(gè)標(biāo)識(shí)符:
trait HairColor
特征作為泛型類型尤其有用,并且?guī)в谐橄蠓椒ā?/p>
trait Iterator[A] {
def hasNext: Boolean
def next(): A
}
擴(kuò)展trait Iterator[A]需要提供類型A,并且要實(shí)現(xiàn)hasNext和next方法。
使用特征
使用extends關(guān)鍵字?jǐn)U展特征。然后使用override關(guān)鍵字實(shí)現(xiàn)特征的任意抽象成員。
trait Iterator[A] {
def hasNext: Boolean
def next(): A
}
class IntIterator(to: Int) extends Iterator[Int] {
private var current = 0
override def hasNext: Boolean = current < to
override def next(): Int = {
if (hasNext) {
val t = current
current += 1
t
} else 0
}
}
val iterator = new IntIterator(10)
iterator.next() // prints 0
iterator.next() // prints 1
IntIterator類有一個(gè)參數(shù)作為上邊界值。擴(kuò)展了Iterator[Int],next方法必須返回Int類型的值。
子類型
可以在需要會(huì)用特征的地方使用其子類型。
import scala.collection.mutable.ArrayBuffer
trait Pet {
val name: String
}
class Cat(val name: String) extends Pet
class Dog(val name: String) extends Pet
val dog = new Dog("Harry")
val cat = new Cat("Sally")
val animals = ArrayBuffer.empty[Pet]
animals.append(dog)
animals.append(cat)
animals.foreach(pet => println(pet.name)) // Prints Harry Sally
trait Pet有一個(gè)抽象字段name,Cat和Dog在其構(gòu)造函數(shù)中實(shí)現(xiàn)了。最后一行,調(diào)用了pet.name,在特征Pet的任意子類型中都必須實(shí)現(xiàn)。