在Scala中,如何使用多个构造函数对Java类进行子类化?

时间:2010-07-21 13:35:12

标签: java scala constructor multiple-constructors

假设我有一个包含多个构造函数的Java类:

class Base {
    Base(int arg1) {...};
    Base(String arg2) {...};
    Base(double arg3) {...};
}

如何在Scala中扩展它并仍然提供对Base的所有三个构造函数的访问?在Scala中,子类只能调用其中一个超类的构造函数。我该如何解决这条规则?

假设Java类是我无法更改的遗留代码。

4 个答案:

答案 0 :(得分:89)

很容易忘记一个特质可能会扩展一个类。如果使用特征,则可以推迟决定调用哪个构造函数,如下所示:

trait Extended extends Base {
  ...
}

object Extended {
  def apply(arg1: Int) = new Base(arg1) with Extended
  def apply(arg2: String) = new Base(arg2) with Extended
  def apply(arg3: Double) = new Base(arg3) with Extended
}

Traits本身可能没有构造函数参数,但您可以通过使用抽象成员来解决这个问题。

答案 1 :(得分:5)

编辑 - 这是来自scala邮件列表上的一个问题,我认为这里的问题是重复的。我的答案涉及提供三种不同的构造函数(即复制Java设计),而不是扩展类

假设每个构造函数最终创建对象的 state S,使用“static”方法创建一个伴随对象来创建此状态

object Base {
  private def stateFrom(d : Double) : S = error("TODO")
  private def stateFrom(s : Str) : S = error("TODO")
  private def stateFrom(i : Int) : S = error("TODO")
} 

然后创建一个私有构造函数,使用状态和(公共)重载构造函数,这些构造函数遵循主构造函数

import Base._
class Base private(s : S) { //private constructor takes the state
  def this(d : Double) = this(stateFrom(d)) 
  def this(str : String) = this(stateFrom(str))
  def this(i : Int) = this(stateFrom(i))
  //etc
}

答案 2 :(得分:2)

我会选择最通用的(在本例中为String)并在符合其他条件的情况下自行进行内部转换。

虽然我承认这不是最好的解决方案,但有些事情让我觉得错误。 : - (

答案 3 :(得分:2)

这是一个愚蠢的答案,可能会有所作为但如果Java类有太多构造函数可能会付出太多努力,但是:

在Java中编写一个实现构造函数的子类,该构造函数接受各种其他构造函数所需的所有输入,并根据输入的存在与否来调用其超类的正确构造函数(通过使用“null”或某种类型的sentinel然后,在Scala中继承Java类,并将sentinel值指定为默认参数。