2011-01-28 17 views
10

Chcę stworzyć cechę, która dodaje pewne właściwości do klasy i umożliwia łańcuch metod. Testowane w Scali 2.8.1.Najlepsza praktyka implementacji cechy Scala, która obsługuje łańcuchowanie metod.

trait SomeProperty { 
    var prop : String = "default" 
    def setProp(s: String) = { 
     prop = s 
     this 
    } 
} 
sealed abstract class Value 
case class IntegerValue(v: Int) extends Value 
case class FloatValue(v: Float) extends Value with SomeProperty { 
    def foo() = { println("I'm foo.") } 
} 
case object UnknownValue extends Value with SomeProperty { 
    def bar() = { println("I'm bar.") } 
} 

scala> val x = UnknownValue 
scala> x.setProp("test").bar() 
<console>:10: error: value bar is not a member of SomeProperty 
    x.setProp("test").bar() 

Jaka jest najczęstsza praktyka w tego rodzaju sytuacjach? (Preferowana metoda zapisu)

Odpowiedz

20

Można jawnie określić typ instancji jako typ powrotu dla metody setProp.

trait SomeProperty { 
    var prop : String = "default" 
    def setProp(s: String):this.type = { 
     prop = s 
     this 
    } 
} 
+0

Jest lepiej. – lscoughlin

+0

To działa! Ale nie rozumiem, dlaczego. Domyślam się, że 'ten.type' jest równy' SomeProperty', czyż nie? Czy ta Scala jest specyficzna? A może także w Javie? – ihji

+0

Ten artykuł jest pomocny. http://scalada.blogspot.com/2008/02/thistype-for-chaining-method-calls.html – ihji

0

Najprostszą rzeczą do zrobienia jest użycie generycznych.

object Value { 

    trait SomeProperty[X] { 
    var str: String = null; 
    def setStr(s: String): X = { 
     str = s; 
     return this.asInstanceOf[X] 
    } 
    } 

    abstract sealed class Value 
    case class IntegerValue(i: Int) 
    case class StringValue(s: String) extends SomeProperty[StringValue] { 
    def foo(): Unit = { 
     println("Foo.") 
    } 
    } 
    case class UnknownValue(o: Any) extends SomeProperty[UnknownValue] { 
    def bar(): Unit = { 
     println("Bar.") 
    } 
    } 

    def main(args: Array[String]): Unit = { 

    new UnknownValue(18).setStr("blah blah blah").bar 
    new StringValue("A").setStr("halb halb halb").foo 
    } 
} 
1

Nie wiem, czy to jest to, czego szukasz

scala> trait Property[T] { 
    | me: T => 
    | var prop:String="" 
    | def setProp(s:String) = { 
    |  prop=s 
    |  me 
    | } 
    | } 
defined trait Property 

scala> class A extends Property[A] 
defined class A 

scala> class B extends Property[B] 
defined class B 

scala> val a= new A 
a: A = [email protected] 

scala> val b = new B 
b: B = [email protected] 

scala> a.setProp("Hi") 
res13: Property[A] with A = [email protected] 

scala> a.setProp("Hi").setProp("Bye") 
res14: Property[A] with A = [email protected] 

scala> b.setProp("D") 
res15: Property[B] with B = [email protected]