假设我有:
val number:AnyVal
我知道 x 可以是任何数字(就我们的目的而言,是 Float、Double、Int、Long)。
将此类数字转换为 Long 的最简单方法是什么:
val l = number.toLong //fails for AnyVal
假设我有:
val number:AnyVal
我知道 x 可以是任何数字(就我们的目的而言,是 Float、Double、Int、Long)。
将此类数字转换为 Long 的最简单方法是什么:
val l = number.toLong //fails for AnyVal
如果您知道它肯定会是浮点数、双精度数、整数或长整数,则可以将其转换为数字。然后你可以调用longValue:
val number:AnyVal = 10
val l:Long = number.asInstanceOf[Number].longValue
怎么样:
scala> import scala.util.Try
import scala.util.Try
scala> val i1: Int = 23
i1: Int = 23
scala> val l1: Long = 42
l1: Long = 42
scala> val f1: Float = 14.9f
f1: Float = 14.9
scala> val d1: Double = 14.96
d1: Double = 14.96
scala> val b1: Boolean = true
b1: Boolean = true
scala> List(i1, l1, f1, d1, b1) map (x => Try(x.asInstanceOf[Number].longValue)) foreach (println(_))
Success(23)
Success(42)
Success(14)
Success(14)
Failure(java.lang.ClassCastException: java.lang.Boolean cannot be cast to java.lang.Number)
scala> List(i1, l1, f1, d1, b1) map (x => Try(x.asInstanceOf[Number].longValue)) foreach (n => println(n.get))
23
42
14
14
java.lang.ClassCastException: java.lang.Boolean cannot be cast to java.lang.Number
at $anonfun$1$$anonfun$apply$1.apply$mcJ$sp(<console>:14)
at $anonfun$1$$anonfun$apply$1.apply(<console>:14)
at $anonfun$1$$anonfun$apply$1.apply(<console>:14)
at scala.util.Try$.apply(Try.scala:161)
at $anonfun$1.apply(<console>:14)
at $anonfun$1.apply(<console>:14)
at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:244)
at scala.collection.TraversableLike$$anonfun$map$1.apply(TraversableLike.scala:244)
at scala.collection.immutable.List.foreach(List.scala:318)
at scala.collection.TraversableLike$class.map(TraversableLike.scala:244)
at scala.collection.AbstractTraversable.map(Traversable.scala:105)
at .<init>(<console>:14)
Scala 2.11 的更新答案
我原来的答案在新版本的 Scala 中不起作用,因为到 RichLong 的隐式转换不再可用。
此更新版本适用于通过 Number 子类型的类型匹配的数字,也适用于通过 Scala 2.11 中的隐式转换的字符串:
object LongNumber {
def cast(number: Any): Long = number match {
case n: Number => n.longValue()
case x => throw new IllegalArgumentException(s"$x is not a number.")
}
// Test cases
def main(args: Array[String]): Unit = {
val twelveByte: Byte = 0x0c
val twelveString: String = "12"
println(s"Converting a long: ${cast(12L)}")
println(s"Converting an int: ${cast(12)}")
println(s"Converting a double: ${cast(12.0)}")
println(s"Converting a byte: ${cast(twelveByte)}")
println(s"Converting a string: $twelveString")
}
}
匹配技术是其他答案中使用的铸造技术的微小变化。
旧版本 Scala 的原始答案
在匹配块中尝试隐式转换为 RichLong 似乎效果很好:
import scala.runtime.RichLong
...
def cast(number: Any): Long = number match {
case n: RichLong => n.toLong
case x => throw new IllegalArgumentException(s"$x is not a number.")
}
如果您想满足这种可能性,也可以添加一个匹配数字格式字符串的案例。
如果您没有关于更具体类型的任何信息,那么您必须对每个选项进行模式匹配。