object CaseTest {
def main(args:Array[String]){
println(List(1,3,5,"seven") map{case i:Int =>i+1})
}
}java
运行结果:
Exception in thread "main" scala.MatchError: seven (of class java.lang.String)
at CaseTest$.$anonfun$main$1(CaseTest.scala:5)
at CaseTest$.$anonfun$main$1$adapted(CaseTest.scala:5)
at scala.collection.immutable.List.map(List.scala:287)
at CaseTest$.main(CaseTest.scala:5)
at CaseTest.main(CaseTest.scala)app
从新修改代码:
println(List(1,3,5,"seven") collect{case i:Int => i+1})
再次执行结果以下:
List(2, 4, 6)
为何将map换成了collect以后就能够成功执行呢?这是由于map函数接受一个普通的匿名函数,当适用于"seven"元素时出现类型匹配错误,而collect接受一个偏函数PartialFunction。case语句在scala中除了能够被编译为匿名函数外,还能够编译为一个偏函数PartialFunction,上面程序中编译为了PartialFunction.由上面执行的结果来分析:偏函数它只对会做用于指定类型的参数或指定范围的参数实施做用,超出它的界定范围以外的参数类型和值它会忽略。经过查看scala中的map和collect的源代码能够看到:
final override def map[B, That](f: A => B)(implicit bf: CanBuildFrom[List[A], B, That]): That = {....}ide
final override def collect[B, That](pf: PartialFunction[A, B])(implicit bf: CanBuildFrom[List[A], B, That]): That = {...}函数
trait PartialFunction[-A, +B] extends (A => B) { self => //看不懂源代码中的这一行中的self =>???????也看不懂[-A,+B]?????
import PartialFunction._ ui
def isDefinedAt(x: A): Booleanthis
def orElse[A1 <: A, B1 >: B](that: PartialFunction[A1, B1]): PartialFunction[A1, B1] =
new OrElse[A1, B1] (this, that)
//TODO: why not overload it with orElse(that: F1): F1?scala
override def andThen[C](k: B => C): PartialFunction[A, C] =
new AndThen[A, B, C] (this, k)ci
def lift: A => Option[B] = new Lifted(this)it
def applyOrElse[A1 <: A, B1 >: B](x: A1, default: A1 => B1): B1 =
if (isDefinedAt(x)) apply(x) else default(x)io
def runWith[U](action: B => U): A => Boolean = { x => val z = applyOrElse(x, checkFallback[B]) if (!fallbackOccurred(z)) { action(z); true } else false } }