Scala概述(五)抽象(2.3)

族多态和self类型(Family polymorphism and self types.Scala的抽象类型概念非常适合于描述相互之间协变的一族(families)类型,这种概念称作族多态。例如:考虑发布/订阅模式,它有两个主要类型:subjectsobserversSubjects定义了subscribe方法,用于给observers进行注册,同时还有一个publish方法,用于通知所有的注册者;通知是通过调用所有注册者的notify方法实现的。一般来说,当subject的状态发生改变时,会调用publish方法。一个subject可以有多个observers,一个observer也可以观察多个subjectSubscribe方法一般用observer的标识为参数,而notify方法则以发出通知的subject对象为参数。因此,这两个类型在方法签名中都引用到了对方。

这个模式的所有要素都在如下系统中:

abstract class SubjectObserver {

type S <: Subject

type O <: Observer

abstract class Subject requires S {

private var observers: List[O] = List()

def subscribe(obs: O) =

observers = obs :: observers

def publish =

for (val obs <- observers) obs.notify(this)

}

 

trait Observer {

def notify(sub: S): unit

}

}

顶层的SubjectObserver类包含两个类成员:一个用于subject,一个用于observerSubject类定义了subscribe方法和publish方法,并且维护一个所有注册的observer的列表。Observer这个trait只定义了一个抽象方法notify

需要注意的是,SubjectObserver并没有直接引用对方,因为这种“硬”引用将会影响客户代码对这些类进行协变的扩展。相反,SubjectOberver定义了两个抽象类型SO,分别以SubjectObserver作为上界。Subjectobserver的类型分别通过这两个抽象类型引用对方。

另外还要注意,Subject类使用了一个特殊的标注requires

abstract class Subject requires S { ...

这个标注表示Subject类只能作为S的某个子类被实例化,这里S被称作Subjectself-type。在定义一个类的时候,如果指定了self-type,则这个类定义中出现的所有this都被认为属于这个self-type类型,否则被认为是这个类本身。在Subject类中,必须将self-type指定为S,才能保证obs.notify(this)调用类型正确

Self-type可以是任意类型,并不一定与当前正在定义的类型相关。依靠如下两个约束,类型正确性仍然可以得到保证:(1)一个类型的self-type必须是其所有父类型的子类,(2)当使用new 对一个类进行实例化时,编译器将检查其self-type必须是这个类的父类。

这个publish/subscribe模式中所定义的机制可以通过继承SubjectObserver,并定义应用相关的SubjectObserver类来使用。例如下面的SensorReader对象,将传感器(sensors)作为subjects,而将显示器(displays)作为observers

object SensorReader extends SubjectObserver {

type S = Sensor

type O = Display

abstract class Sensor extends Subject {

val label: String

var value: double = 0.0

def changeValue(v: double) = {

value = v

publish

}

}

 

class Display extends Observer {

def println(s: String) = ...

def notify(sub: Sensor) =

println(sub.label + " has value " + sub.value)

}

}

在这个对象中,SSensor限定,而ODisplay限定,从而原先的两个抽象类型现在分别通过覆盖而获得定义,这种“系绳节”(“tying the knot”)在创建对象实例的时候是必须的。当然,用户也可以再定义一个抽象的SensorReader类型,未来再通过继承进行实例化。此时,这两个抽象类型也可以通过抽象类型来覆盖,如:

class AbsSensorReader extends SubjectObserver {

type S <: Sensor

type O <: Display

...

}

下面的代码演示了SensorReader如何使用:

object Test {

import SensorReader._

val s1 = new Sensor { val label = "sensor1" }

val s2 = new Sensor { val label = "sensor2" }

def main(args: Array[String]) = {

val d1 = new Display; val d2 = new Display

s1.subscribe(d1); s1.subscribe(d2)

s2.subscribe(d1)

s1.changeValue(2); s2.changeValue(3)

}

}

另外值得注意的是其中的import语句,它使Test可以直接访问SensorReader的成员,而无需前缀。ScalaImportJava中用法更广泛,可以在任何地方使用,可以从任何对象中导入成员,而不仅仅从一个package中。

你可能感兴趣的:(scala)