首页 > 代码库 > scala学习手记39 - 模式匹配

scala学习手记39 - 模式匹配

在java中有switch/case这样的模式匹配语句,可以匹配的类型包括int,byte,char,short, enum,在java8又支持了字符串。

在scala中也有类似的模式匹配语句,即match-case。这个好现在之前使用过一次。scala中的match-case匹配的类型更为广泛,它是对Any类型起作用的。

来看个例子:

def activity(day: String) {  day match {    case "Sunday" => println("Eat, sleep, repeat... ")    case "Saturday" => println("Hangout with friends... ")    case "Monday" => println("...code for fun...")    case "Friday" => println("...read a good book...")  }}List("Monday", "Sunday", "Saturday").foreach {  activity}

执行结果:

技术分享

有没有注意到,这里和java是有些不一样的,最大的区别是没有break。但是在匹配上了后,也不会继续顺序执行了。其实我一直觉得java的那个switch/case在这一点上是有些不足的,这次总算是在scala中解决了。不过这里也有些让人不爽的地方:比如没有匹配到就会抛出异常。不管怎样,这个异常是不能随意抛出的。在Java的switch/case中,是有一个default来处理那些没有匹配到的内容的。在scala中也有一个特殊的符号起到了类似的作用,就是“_”,这里的下划线是一个通配符。继续修改下上面的代码:

def activity(day: String) {  day match {    case "Sunday" => println("Eat, sleep, repeat... ")    case "Saturday" => println("Hangout with friends... ")    case "Monday" => println("...code for fun...")    case "Friday" => println("...read a good book...")    case _ => println("...nothing...")  }}List("Monday", "Sunday", "Saturday", "Tuesday").foreach {  activity}

就不贴执行结果了。

需要注意的是:case表达式并不限于在match语句里使用。这里,包含case表达式的代码块就是一个简单函数值。

刚才说过match-case支持的类型是Any。在java支持的基本类型、枚举和字符串之外,scala也支持其他任意类型的实例,比如列表和元组:

def processCoordinates(input: Any) {  input match {    case (a, b) => printf("Processing (%d, %d)... ", a, b)    case List("red", "blue", "white") => println("Stars and Stripes...")    case List("red", "blue", _*) => println("colors red, blue, ... ")    case "done" => println("done")    case _ => null  }}processCoordinates((39, -104))processCoordinates("done")processCoordinates(List("red", "blue", "green"))processCoordinates (List("red", "blue", "white"))

看下执行结果:

技术分享

前面有一段警告,可以不去管它。

不过代码里却是有些不足之处,其中对元组的匹配还可以更细致些,比如针对不同类型的值可以做不同的处理:

def process(input: Any) {  input match {  case (a: Int, b: Int) => print("Processing (int, int)... ") -  case (a: Double, b: Double) => print("Processing (double, double)... ")  case msg: Int if (msg > 1000000) => println("Processing int > 1000000")  case msg: Int => print("Processing int... ")  case msg: String => println("Processing string... ") -  case _ => printf("Can‘t handle %s... ", input)  }}process ((34.2, -159.3))process(0)process(1000001)process (2.2)

代码中演示了在case语句里如何为单一的值和元组元素指定类型。除了类型之外,还可以使用卫述句(guard)。卫述句用if从句表示,在模式匹配里,对表达式求值前必须满足卫述句。

看下结果:

技术分享

case的顺序很重要。Scala会自上而下地求值。所以,上面代码5和6两行是不能交换的。

##########

scala学习手记39 - 模式匹配