在调用this()之前在重载的构造函数中执行代码

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成为第一个语句.

那么这个问题的解决方案是什么?

Did*_*ont 6

def this(url: Url) = this(try {XML.load(url)} catch {case _ => <results/>})
Run Code Online (Sandbox Code Playgroud)

更一般地说,参数的评估必须在构造函数调用之前进行,所以你在那里做(scala中的一个块是一个表达式,但写一个例程,通常写在伴随对象中,如果它太长了).你不能做的是让这段代码选择你调用的其他构造函数.但是,由于他们所有人必须前往主要人员,所以你不会损失太多.此外,您需要调用的其他构造函数至少有一个参数.如果有几个构造函数,那么主要的构造函数通常不应该是没有参数的构造函数(请参阅Scala问题可选构造函数)


par*_*tic 5

伴随对象中的工厂方法:

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关键字.