10

是否可以在 Scala 构造函数中使用命名参数,然后在不破坏构造函数接口或使代码非常难看的情况下覆盖 getter 和 setter?

获取以下 scala 代码

class Person( var FirstName: String, var LastName: String )

干净整洁。这将创建一个名为 person 的简单类,我们可以通过以下方式使用它

val john = new Person( FirstName="John", LastName="Doe" )
john.FirstName = "Joe"
println( john.FirstName )

稍后,我们决定要向 FirstName 设置器添加一些验证。因此,我们创建一个新的私有局部变量并覆盖 getter 和 setter 方法

class Person( var _FirstName: String, var _LastName: String ) {

    def FirstName = _FirstName  
    def FirstName_= (value:String) = _FirstName = value

}

仍然有些干净,但是为了做到这一点,我们不得不更改构造函数参数名称,从而破坏了外部接口。

我想出的第一个解决这个问题的方法是

class Person {
    var _FirstName:String = null 
    var LastName:String  = null

    def FirstName = _FirstName  
    def FirstName_= (value:String) = _FirstName = value

    def this( FirstName: String, LastName: String ){
        this()
        this._FirstName = FirstName
        this.LastName = LastName 
    }

}

这有点丑陋和不雅,并且消除了我最初使用 scala 的大部分好理由。

有没有更好的方法来做到这一点?

tl;dr 如何覆盖默认构造函数中定义的成员的 getter/setter 而不使代码丑陋或更改公共接口?

4

3 回答 3

10

你考虑过使用伴生对象吗?

class Person private (f: String, l: String ) {
   var FirstName = f
   var LastName = l
}

object Person {
   def apply(FirstName:String, LastName:String) = 
       new Person(FirstName, LastName) 
}
于 2011-02-21T15:07:45.243 回答
5

如果您尚未使用隐式转换来创建参数,则可以执行以下操作:

def validateName(s: String) = {
  if (s.length>0 && s(0).isUpper) s
  else throw new IllegalArgumentException(s+" is not a name!")
}

object Example {
  private[Example] class ValidatedName(val s: String) { }
  class Person(var firstName: ValidatedName, var lastName: String) { }
  implicit def string2valid(s: String) = new ValidatedName(validateName(s))
  implicit def valid2string(v: ValidatedName) = v.s
}

scala> new Example.Person("Joe","Schmoe")
res17: Example.Person = Example$Person@51887dd5

scala> new Example.Person("ee","cummings")
java.lang.IllegalArgumentException: ee is not a name!

它不是二进制兼容的,但它是源兼容的(同样,如果名称尚未依赖隐式转换)。

另一个稍长一点的可能性是创建一个隐形祖先:

class CheckedPerson(private var first: String, var lastName: String) {
  def firstName = first
  def firstName_=(s: String) { first = validateName(s) }
}
class Person(firstName: String, lastName: String) extends
  CheckedPerson(validateName(firstName),lastName) { }

我不确定二进制兼容性,但肯定会提供源兼容性。

于 2011-02-21T16:23:08.740 回答
4

不,目前没有办法做到这一点,这不是目前研究的重点。

这是我对该语言的主要不满之一:没有明智的方法将构造函数参数和自定义的 getter/setter 方法结合起来。

如果您对所提供的功能不满意class Person( var FirstName: String, var LastName: String ),它基本上意味着“回到 Java 的冗长性”。

于 2011-02-21T13:51:09.213 回答