我正在反思性地调用一个方法,该方法的参数可能是也可能不是值类的实例。由于值类的目的是避免基础值的装箱,如果参数类型是值类,则相关方法实际上需要未装箱的值。为了处理这种情况,我正在尝试从值类中解开基础值。我首先需要确定参数是否属于值类,在这里我遇到了第一个绊脚石:
def isObjectOfValueClass(arg: Any) =
classOf[AnyVal].isAssignableFrom(arg.getClass)
这不会按预期工作,因为该方法返回以下true
:
case class NonValueClass(underlying: Int)
如何实施isObjectOfValueClass
?或者有没有更简单的方法来反射性地调用可能将值类的对象作为参数的方法?
首先,请注意,您的isObjectOfValueClass
将获得值类实例的盒装版本。
其次,它不能像你想要的那样工作。这是因为classOf[AnyVal] == classOf[AnyRef] == <java.lang.Object>
.
没有运行时方法来区分盒装值类和引用类(Any
没有.instanceOf[T]
,AnyVal
不能用于模式匹配或作为.instanceOf[T]
的参数,最重要的是,编译的值类不会扩展或实现AnyVal
)。
如果您希望在编译时决定它,请尝试:
case class IsAnyVal[-T](val value: Boolean) extends AnyVal
implicit def _noClueHowToNameThisImplicit_1 = IsAnyVal[AnyVal](true)
implicit def _noClueHowToNameThisImplicit_2 = IsAnyVal[AnyRef](false)
def isAnyVal[T](arg: T)(implicit ev: IsAnyVal[T]) = ev.value
scala> isAnyVal(1)
res4: Boolean = true
scala> isAnyVal("")
res5: Boolean = false
我不确定您希望如何在没有更多意外装箱的情况下提取检测到的盒装值类实例的唯一字段。此外,Hotspot非常擅长优化小型短期对象。