Scala - 在不对其进行子类化的情况下扩展引用类型

时间:2014-11-11 16:18:49

标签: java scala

我有一个我想要子类化的Java类。子类只是添加便利方法 - 它可以全部使用外部方法完成,因为我只查看公共字段,并且我不会修改任何内容。

如果基类是值类型,我使用值包装器 - extends AnyVal。但是基类是Java引用类型。除了扩展它之外,有没有更好的方法来进行子类化?

2 个答案:

答案 0 :(得分:10)

要专门解决第二段,使用值类包装的类型可以作为引用类型,并且仍然可以避免通常在包装中涉及的额外对象分配。例如,如果你有这些隐式类:

implicit class MyInt(val underlying: Int) extends AnyVal {
  def inc: Int = underlying + 1
}

implicit class MyString(val underlying: String) extends AnyVal {
  def firstChar: Char = underlying.charAt(0)
}

implicit class MyNonValueClassString(val underlying: String) {
  def firstCharNonValueClass: Char = underlying.charAt(0)
}

这个使用它们的代码:

println(42.inc)
println("hello".firstChar)
println("hello".firstCharNonValueClass)

您可以使用-Xprint:flatten进行编译以查看desugared版本(为清晰起见,此处重新格式化):

scala.this.Predef.println(
  scala.Int.box(Demo$MyInt.this.inc$extension(Demo.this.MyInt(42)))
);

scala.this.Predef.println(
  scala.Char.box(
    Demo$MyString.this.firstChar$extension(Demo.this.MyString("hello"))
  )
);

scala.this.Predef.println(
  scala.Char.box(
    Demo.this.MyNonValueClassString("hello").firstCharNonValueClass()
  )
);

如您所见,firstChar调用不涉及新对象。

答案 1 :(得分:2)

赞成合成而不是继承。

因此,在您的情况下,最好的方法可能是创建一个将Java类作为属性的类。然后你可以简单地委托这些对象的方法并添加你自己的方法:

class MyClass(jo: JavaObject) {
    def delegateMethod() = jo.method()
    def newMethod() = // ...
}