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两行是不能交换的。
##########
仅是学习笔记,难免出错,望不吝指点