Nik*_*kov 5 constructor scala overloading
假设我们有这样一个类:
import java.net.URL
import xml._
class SearchData(xml: Node) {
def this(url: URL) = this (XML.load(url))
}
Run Code Online (Sandbox Code Playgroud)
我们想在调用之前执行一些代码this (XML.load(url))- 比如用它来测试它try.人们会期望写这样的东西会起作用:
class SearchData(xml: Node) {
def this(url: URL) {
try {
this (XML.load(url))
} catch {
case _ => this(<results/>)
}
}
}
Run Code Online (Sandbox Code Playgroud)
但它不会,因为Scala要求您this()在重载的构造函数中调用第一个语句,在这种情况下try成为第一个语句.
那么这个问题的解决方案是什么?
def this(url: Url) = this(try {XML.load(url)} catch {case _ => <results/>})
Run Code Online (Sandbox Code Playgroud)
更一般地说,参数的评估必须在构造函数调用之前进行,所以你在那里做(scala中的一个块是一个表达式,但写一个例程,通常写在伴随对象中,如果它太长了).你不能做的是让这段代码选择你调用的其他构造函数.但是,由于他们所有人必须前往主要人员,所以你不会损失太多.此外,您需要调用的其他构造函数至少有一个参数.如果有几个构造函数,那么主要的构造函数通常不应该是没有参数的构造函数(请参阅Scala问题可选构造函数)
伴随对象中的工厂方法:
object SearchData {
def apply(xml: Node) = new SearchData(xml) //Added to provide uniform factories
def apply(url: URL) = {
try {
new SearchData(XML.load(url))
} catch {
case _ => new SearchData(<results/>)
}
}
}
//Example
val sd = SearchData( new URL( "http://example.com/" ) )
Run Code Online (Sandbox Code Playgroud)
它不仅简化了设计,还可以省去new关键字.