Value Classes 的功能和隐式转换很像,优点是在运行时拥有非常小的内存开销。所有的 Value Classes 都继承自 AnyVal ,例如:
class Wrapper(val underlying: Int) extends AnyVal
Value Classes 主要用于扩展已有类的方法和创建类型安全的数据类型。
扩展已有类的方法
结合 implicit classes ,我们可以利用 Value Class 扩展已有类的方法, Scala 基础类库中的 RichInt 就是一个很好的例子,正是因为 RichInt 定义了 toHexString 方法,
implicit class RichInt(val self: Int) extends AnyVal {
def toHexString: String = java.lang.Integer.toHexString(self)
}
所以我们才能直接在 Int 类型上使用这个方法:
3.toHexString
在运行时,上面的代码会转换成静态方法调用,不会创建任何新对象,
RichInt$.MODULE$.extension$toHexString(3)
类型安全的数据类型
Value Class 的另一个用法是创建类型安全的数据类型,并且没有额外的运行时内存开销。例如我们可以创建一个表示距离的 Value Class,
class Meter(val value: Double) extends AnyVal {
def +(m: Meter): Meter = new Meter(value + m.value)
}
然后我们可以对两个距离相加,
val x = new Meter(3.4)
val y = new Meter(4.3)
val z = x + y
在运行时,上面的代码不会创建任何的 Meter 对象,而只是基于 double 类型做基本的数学运算。这样既保证了 Meter 类型在做加法运算时的类型安全,同时也降低了运行时的内存开销。
参考:https://docs.scala-lang.org/overviews/core/value-classes.html