引言

模式匹配是Scala中非常有特色,非常强大的一种功能。

类似于Java中的switch case语法,但是模式匹配的功能要比它强大得多,switch只能对值进行匹配,但是Scala的模式匹配除了可以对值进行匹配之外,还可以对类型进行匹配、对Array和List的元素情况进行匹配、对case class进行匹配甚至对有值或没值(Option)进行匹配

而且在Spark源码中也大量地使用了模式匹配功能。因此为了更好地编写Scala程序,并且更加通畅地看懂Spark源码,学好模式匹配是非常重要的。

基础语法

match case的语法如下:变量 match { case 值 => 代码}。如果值为下划线,则代表不满足以上所有情况下的默认情况如何处理。此外,match case中,只要一个case分支满足并处理了,就不会继续判断下一个case分支了。(这一点与java不同,java的switch case需要用break阻止)

对变量值进行匹配


// 案例:成绩评价
def judgeGrade(grade: String) {
grade match {
case "A" => println("Excellent")
case "B" => println("Good")
case "C" => println("Just so so")
case _ => println("you need work harder")
}
}
judgeGrade: (grade: String)Unit
scala> judgeGrade("E")
you need work harder
scala> judgeGrade("B")
Good

在模式匹配中使用if守卫


// 案例:成绩评价(升级版)
def judgeGrade(name: String, grade: String){
grade match {
case "A" => println("Excellent")
case "B" => println("Good")
case "C" => println("Just so so")
case _ if name == "leo" => println(name + ", you are a good boy, come on")
case _ => println("you need to work harder")
}
}
judgeGrade: (name: String, grade: String)Unit
scala> judgeGrade("leo", "E")
leo, you are a good boy, come on
scala> judgeGrade("sparks", "E")
you need to work harder

在模式匹配中进行变量赋值

模式匹配中可以将默认情况即下划线替换为一个变量名,这样模式匹配语法就会将要匹配的值赋值给这个变量,从而可以在后面的处理语句中使用要匹配的值。


// 案例:成绩评价(升级版)
def judgeGrade(grade: String){
grade match {
case "A" => println("you got A grade, excellent!")
case "B" => println("you got B grade, good")
case "C" => println("you got C grade, so so")
case badGrade => println("you got " + badGrade + "grade, i hope that you can get C next time")
}
}
judgeGrade: (grade: String)Unit
scala> judgeGrade("E")
you got Egrade, i hope that you can get C next time
scala> judgeGrade("F")
you got Fgrade, i hope that you can get C next time

类型匹配

Scala的模式匹配强大之处就在于可以直接匹配类型而不仅仅是值,这也是Java中switch case绝对做不到的。

匹配类型的语法为case 变量 : 类型 => 代码


// 案例:异常处理
import java.io._
def processException(e: Exception){
e match {
case e1:IllegalArgumentException => println("you passed illegal argument, exception is : " +e1)
case e2:IOException => println("you got an error while doing IO operation!" + e2)
case _:Exception => println("cannot know which exception you have")
}
}
processException: (e: Exception)Unit
// 测试能否匹配类型
scala> processException(new IllegalArgumentException("expect two arguments, but found only one"))
you passed illegal argument, exception is : java.lang.IllegalArgumentException: expect two arguments
, but found only one
scala> processException(new ArrayIndexOutOfBoundsException("array is null."))
cannot know which exception you have

对Array和List的元素进行匹配

对Array进行模式匹配,分别可以匹配带有指定元素的数组、带有指定个数的数组、以某元素打头的数组。


def greeting(arr: Array[String]) {
arr match {
case Array("Leo") => println("Hi, Leo!")
case Array(girl1,girl2,girl3) => println("Hi, girls, may I know you name? "+girl1+" "+girl2+" "+
girl3)
case Array("Leo", _*) => println("Hi, leo, please introduce your friends to me.")
case _ => println("hey, who are you?")
}
}
greeting: (arr: Array[String])Unit
// 测试
scala> greeting(Array("Leo"))
Hi, Leo!
scala> greeting(Array("jen", "Alice", "lory"))
Hi, girls, may I know you name? jen Alice lory
scala> greeting(Array("sparks"))
hey, who are you?

  
对List进行模式匹配,与Array类似,但是需要使用List特有的 :: 操作符。


def greeting(list: List[String]){
list match {
case "Leo" :: Nil => println("Hi, Leo!")
case girl1 :: girl2 :: girl3 ::Nil => println("Hi, girls, nice to meet you." + girl1 + " " + gir
l2 + " " + girl3)
case "Leo" :: tail => println("Hi, Leo, please introduce your friends to me.")
case _ => println("hey, who are you?")
}
}
greeting: (list: List[String])Unit
// 测试
scala> greeting(List("Leo"))
Hi, Leo!
scala> greeting(List("Marry", "Alice", "lory"))
Hi, girls, nice to meet you.Marry Alice lory
scala> greeting(List("Sparks"))
hey, who are you?

case class匹配

Scala中提供了一种特殊的类,用case class进行声明,中文也可以称作样例类。case class其实有点类似于Java中的JavaBean的概念。即只定义field,并且由Scala编译时自动提供getter和setter方法,但是没有method

case class的主构造函数接收的参数通常不需要使用val或var修饰,Scala自动就会使用val修饰(但是如果你自己使用var修饰,那么还是会按照var来定义)

Scala自动为case class定义了伴生对象,也就是object,并且定义了apply()方法,该方法接收主构造函数中相同的参数,并返回case class对象。


// 定义case class
class Person
case class Teacher(name: String, subject: String) extends Person
case class Student(name: String, classroom: String) extends Person
case class Worker(name: String) extends Person
// 案例:学校门禁
def judgeIdentify(p: Person) {
p match {
case Teacher(name, subject) => println("Teacher, name is " +name+ ",subject is " + subject)
case Student(name, classroom) => println("Student, name is" + name + ", classroom is " + classro
om)
case _ => println("Illegal access, please go out of the school")
}
}
defined class Person
defined class Teacher
defined class Student
defined class Worker
judgeIdentify: (p: Person)Unit
// 测试匹配效果
scala> val leo: Person = Student("leo", "class1")
leo: Person = Student(leo,class1)
scala> val tom: Person = Teacher("tom", "Math")
tom: Person = Teacher(tom,Math)
scala> val jack: Person = Worker("jack")
jack: Person = Worker(jack)
scala> judgeIdentify(leo)
Student, name isleo, classroom is class1
scala> judgeIdentify(tom)
Teacher, name is tom,subject is Math
scala> judgeIdentify(jack)
Illegal access, please go out of the school

Option匹配

Scala中有一种特殊的类型,叫做Option。Option有两种值,一种是Some,表示有值,一种是None,表示没有值。

Option通常会用于模式匹配中,用于判断某个变量是有值还是没有值,这比null来的更加简洁明了。

Spark源码中大量地使用了Option,比如Some(a)、None这种语法。


// 案例:成绩查询
val grades = Map("Leo" -> "A", "Jack" -> "B", "Sparks" -> "C")
def getGrade(name: String) {
val grade = grades.get(name)
grade match {
case Some(grade) => println("your grade is " + grade)
case None => println("Sorry, your grade information is not in the system")
}
}、
grades: scala.collection.immutable.Map[String,String] = Map(Leo -> A, Jack -> B, Sparks -> C)
getGrade: (name: String)Unit
// 测试
scala> getGrade("Leo")
your grade is A
scala> getGrade("Sparks")
your grade is C
scala> getGrade("haha")
Sorry, your grade information is not in the system

Scala入门系列(十一):模式匹配的更多相关文章

  1. Scala入门系列(一):基础语法

    Scala基础语法 Scala与JAVA的关系 Scala是基于Java虚拟机,也就是JVM的一门编程语言,所有Scala的代码都需要经过编译为字节码,然后交由Java虚拟机来运行. 所以Scala和 ...

  2. Scala入门系列(四):Map & Tuple

    Map 创建Map // 创建一个不可变的Map scala> val ages = Map("Leo" -> 30, "Sparks" -> ...

  3. Scala入门系列(五):面向对象之类

    定义类 // 定义类,包含field以及method class HelloWorld { private var name = "Leo" def sayHello() { pr ...

  4. Scala入门系列(七):面向对象之继承

    extends 与Java一样,也是使用extends关键字,使用继承可以有效复用代码 class Person { private var name = "leo" def ge ...

  5. Scala入门系列(八):面向对象之trait

    基础知识 1 将trait作为接口使用 此时Trait就与Java中的接口非常类似,不过注意,在Scala中无论继承还是trait,统一都是extends关键字. Scala跟Java 8前一样不支持 ...

  6. Scala入门系列(九):函数式编程

    引言 Scala是一门既面向对象,又面向过程的语言,Scala的函数式编程,就是Scala面向过程最好的佐证.也真是因此让Scala具备了Java所不具备的更强大的功能和特性. 而之所以Scala一直 ...

  7. Scala入门系列(十):函数式编程之集合操作

    1. Scala的集合体系结构 Scala中的集合体系主要包括(结构跟Java相似): Iterable(所有集合trait的根trait) Seq(Range.ArrayBuffer.List等) ...

  8. Scala入门系列(十二):隐式转换

    引言 Scala提供的隐式转换和隐式参数功能,是非常有特色的功能.是Java等编程语言所没有的功能.它可以允许你手动指定,将某种类型的对象转换成其他类型的对象.通过这些功能可以实现非常强大而且特殊的功 ...

  9. Scala入门系列(十三):类型参数

    引言 Scala中类型参数是什么呢?其实就类似于Java中的泛型.定义一种类型参数,比如在集合.类.函数中定义类型参数,然后就可以保证使用到该类型参数的地方就只能是这种类型,从而实现程序更好的健壮性. ...

随机推荐

  1. linux学习(二)linux配置网卡以及常见网络问题排查

    实验环境环境:mac,vmware fusion 一.常用的虚拟机网络连接模式. NAT:推荐方式.它可以使你在切换网络环境(比如在工作中和家里)时,不需要修改虚拟主机的配置,而维持正常的上网功能. ...

  2. 用iptables 做NAT代理上网

    背景:有一台A服务器不能上网,和B服务器通过内网来连接,B服务器可以上网,要实现A服务器也可以上网. 内网主机: A eth1:172.16.1.8 外网主机: B eth0:10.0.0.61外网主 ...

  3. Node.js Stream(流)

    Stream 是一个抽象接口,Node 中有很多对象实现了这个接口.例如,对http 服务器发起请求的request 对象就是一个 Stream,还有stdout(标准输出). Node.js,Str ...

  4. [转]移动前端开发之viewport的深入理解

    今天去面试,被问到一个用了一万次的东西,然而我并不了解具体是个毛毛,看这一篇豁然开朗. DevicePixelRatio 以及这句话:移动设备上的viewport分为layout viewport  ...

  5. HBase流量限制和表负载均衡剖析

    1.概述 在HBase-1.1.0之前,HBase集群中资源都是全量的.用户.表这些都是没有限制的,看似完美实则隐患较大.今天,笔者就给大家剖析一下HBase的流量限制和表的负载均衡. 2.内容 也许 ...

  6. ASP.NET Core 应用程序Startup类介绍

    Startup类配置服务和应用程序的请求管道. Startup 类 ASP.NET Core应用程序需要一个启动类,按照惯例命名为Startup.在主程序的Web Host生成器(WebHostBui ...

  7. Json字符串解析原理、超大json对象的解析

    概述 附上完整的代码:https://pan.baidu.com/s/1dEDmGz3(入口类是Json)JSON:JavaScript 对象表示法(JavaScript Object Notatio ...

  8. 判断pdf、word文档、图片等文件类型(格式)、大小的简便方法

    判断pdf.word文档.图片等文件类型(格式).大小的简便方法 很久没发文了,今天有时间就写一下吧. 关于上传文件,通常我们都需要对其进行判断,限制上传的类型,如果是上传图片,我们甚至会把图片转化成 ...

  9. 【Java框架型项目从入门到装逼】第三节 - 如何用Tomcat发布web项目?

    啥叫Tomcat?有道词典是这么说的. 这个我们姑且不管,实际上呢,Tomcat是一种Web服务器,我们自己做好了一个Web项目,就可以通过Tomcat来发布.服务器呢,又分为硬件服务器和软件服务器. ...

  10. laravel查看执行sql的

    加个监听就好了~~~~而且很简单 1.在routes.php(api.php\web.php)中添加如下语句 Event::listen('illuminate.query', function($s ...