Перегрузка конструктора Scala?


Ответы:


186

Стоит прямо упомянуть, что вспомогательные конструкторы в Scala должны либо вызвать первичный конструктор (как в landon9720), либо другой вспомогательный конструктор из того же класса, что и их первое действие. Они не могут просто вызвать конструктор суперкласса явно или неявно, как в Java. Это гарантирует, что основной конструктор является единственной точкой входа в класс.

class Foo(x: Int, y: Int, z: String) {  
  // default y parameter to 0  
  def this(x: Int, z: String) = this(x, 0, z)   
  // default x & y parameters to 0
  // calls previous auxiliary constructor which calls the primary constructor  
  def this(z: String) = this(0, z);   
}

@ Джон Маколифф: Плохой пример? Без второго и третьего конструкторов пользователь по-прежнему может вызывать, new Foo(x=2,z=4)и new Foo(z=5)если вы измените свою первую строку наclass Foo(x: Int = 0, y: Int = 0, z: String) {
user2987828

Именованные / Стандартные аргументы не поступали до Scala 2.8.
Джон Маколифф

2
Стоит упомянуть, как использовать конструктор перегрузки. Нетривиально, что newключевое слово необходимо даже для тематических классов.
Readren


16

Начиная с Scala 2.8.0, вы также можете иметь значения по умолчанию для параметров contructor- и method. Как это

scala> class Foo(x:Int, y:Int = 0, z:Int=0) {                           
     | override def toString() = { "Foo(" + x + ", " + y + ", " + z + ")" }
     | }
defined class Foo

scala> new Foo(1, 2, 3)                                                    
res0: Foo = Foo(1, 2, 3)

scala> new Foo(4)                                                          
res1: Foo = Foo(4, 0, 0)

Параметры со значениями по умолчанию должны идти после тех, у которых нет значений по умолчанию в списке параметров.


3
Это не работает для нетривиальных значений по умолчанию. так class Foo(val x:Int, y:Int=2*x)не работает.
2012 года

@ Йорген Лундберг: вы написали, что параметры со значениями по умолчанию должны идти после тех, у которых нет значений по умолчанию в списке параметров. Это неправильно, new Foo(x=2,z=4)напечатает Foo(2,0,4).
user2987828

@ user2987828 Я имел в виду, что вы не можете написать новый Foo (12, x = 2), вы должны написать новый Foo (x = 2, 12). Вы можете написать новый Foo (12, y = 2), тогда вы получите Foo (12, 2, 0)
Йорген Лундберг

10

Глядя на мой код, я внезапно осознал, что перегружаю конструктор. Затем я вспомнил этот вопрос и вернулся, чтобы дать другой ответ:

В Scala вы не можете перегружать конструкторы, но вы можете сделать это с помощью функций.

Кроме того, многие решили сделать applyфункцию объекта-компаньона фабрикой для соответствующего класса.

Делая этот класс абстрактным и перегружая applyфункцию для реализации-реализации этого класса, у вас есть перегруженный «конструктор»:

abstract class Expectation[T] extends BooleanStatement {
    val expected: Seq[T]}

object Expectation {
    def apply[T](expd:     T ): Expectation[T] = new Expectation[T] {val expected = List(expd)}
    def apply[T](expd: Seq[T]): Expectation[T] = new Expectation[T] {val expected =      expd }

    def main(args: Array[String]): Unit = {
        val expectTrueness = Expectation(true)}
}

Обратите внимание, что я явно определяю каждый из них, applyчтобы он возвращал Expectation[T], иначе он вернул бы тип с уткой Expectation[T]{val expected: List[T]}.


0

Попробуй это

class A(x: Int, y: Int) {
  def this(x: Int) = this(x, x)
  def this() = this(1)
  override def toString() = "x=" + x + " y=" + y
  class B(a: Int, b: Int, c: String) {
    def this(str: String) = this(x, y, str)
    override def toString() =
      "x=" + x + " y=" + y + " a=" + a + " b=" + b + " c=" + c
  }
}
Используя наш сайт, вы подтверждаете, что прочитали и поняли нашу Политику в отношении файлов cookie и Политику конфиденциальности.
Licensed under cc by-sa 3.0 with attribution required.