полагаться на методы case-класса в признаке - PullRequest
16 голосов
/ 17 марта 2011

Есть ли способ полагаться на методы, определенные в классе case в признаке?Например, копирование: следующее не работает.Хотя я не уверен, почему.

trait K[T <: K[T]] {
  val x: String
  val y: String
  def m: T = copy(x = "hello")
  def copy(x: String = this.x, y: String = this.y): T
}

case class L(val x: String, val y: String) extends K[L]

Дает:

error: class L needs to be abstract, since method copy in trait K of type 
(x: String,y: String)L is not defined
           case class L(val x: String, val y: String) extends K[L]
                      ^

Ответы [ 3 ]

14 голосов
/ 18 марта 2011

Решение состоит в том, чтобы объявить, что ваша черта должна быть применена к классу с методом копирования:

trait K[T <: K[T]] {this: {def copy(x: String, y: String): T} =>
  val x: String
  val y: String
  def m: T = copy(x = "hello", y)
}

(к сожалению, вы не можете использовать неявный параметр в методе копирования, так как неявное объявление не разрешено в объявлении типа)

Тогда ваша декларация в порядке:

case class L(val x: String, val y: String) extends K[L]

(проверено в REPL scala 2.8.1)

Причина, по которой ваша попытка не работает, объясняется в решении, предложенном другими пользователями: ваше объявление copy блокирует генерацию метода "case copy".

4 голосов
/ 17 марта 2011

Я полагаю, что наличие метода с именем copy в trait указывает компилятору не генерировать копию метода в классе case - так что в вашем примере копия метода не реализована в вашем классе case. Ниже приведен небольшой эксперимент с копией метода, реализованной в черте:

scala> trait K[T <: K[T]] {                                                                                   
     | val x: String                                                                                          
     | val y: String                                                                                          
     | def m: T = copy(x = "hello")                                                                           
     | def copy(x: String = this.x, y: String = this.y): T = {println("I'm from trait"); null.asInstanceOf[T]}
     | }

defined trait K

scala> case class L(val x: String, val y: String) extends K[L]                                                
defined class L

scala> val c = L("x","y")                                                                                     
c: L = L(x,y)

scala> val d = c.copy()                                                                                       
I'm from trait
d: L = null
1 голос
/ 17 марта 2011

Вы можете запустить repl с помощью $ scala -Xprint: typer. С параметром -Xprint: typer вы можете видеть, что именно происходит при создании черты или класса. И вы увидите из вывода, что метод «copy» не создан, поэтому компилятор просит определить его самостоятельно.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...