12、Scala基础之模式匹配

引言

模式匹配是Scala中非常有特色,非常强大的一种功能。模式匹配,其实类似于Java中的swich case语法,即对一个值进行条件判断,然后针对不同的条件,进行不同的处理。
但是Scala的模式匹配的功能比Java的swich case语法的功能要强大地多,Java的swich case语法只能对值进行匹配。但是Scala的模式匹配除了可以对值进行匹配之外,还可以对类型进行匹配、对Array和List的元素情况进行匹配、对case class进行匹配、甚至对有值或没值(Option)进行匹配。
而且对于Spark来说,Scala的模式匹配功能也是极其重要的,在spark源码中大量地使用了模式匹配功能。因此为了更好地编写Scala程序,并且更加通畅地看懂Spark的源码,学好模式匹配都是非常重要的。

模式匹配

Scala是没有Java中的switch case语法的,相对应的,Scala提供了更加强大的match case语法,即模式匹配,类替代switch case,match case也被称为模式匹配
Scala的match case与Java的switch case最大的不同点在于,Java的switch case仅能匹配变量的值,比1、2、3等;而Scala的match case可以匹配各种情况,比如变量的类型、集合的元素、有值或无值
match case的语法如下:变量 match { case 值 => 代码 }。如果值为下划线,则代表了不满足以上所有情况下的默认情况如何处理。此外,match case中,只要一个case分支满足并处理了,就不会继续判断下一个case分支了。(与Java不同,java的switch case需要用break阻止)
match case语法最基本的应用,就是对变量的值进行模式匹配

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    judgeScore('F')
  }

  def judgeScore(score:Char)={
    score match {
      case 'A' => println("Excellent")
      case 'B' => println("Great")
      case 'C' => println("Just so so")
      case _ => println("you need to work hard")
    }
  }
}

在模式匹配中使用if守卫

Scala的模式匹配语法,有一个特点在于,可以在case后的条件判断中,不仅仅只是提供一个值,而是可以在值后面再加一个if守卫,进行双重过滤

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    judgeScore('F',"Zhao Jun")
    judgeScore('F',"Feng Xiangbin")
  }

  def judgeScore(score:Char, name:String)={
    score match {
      case 'A' => println("Excellent")
      case 'B' => println("Great")
      case 'C' => println("Just so so")
      case _ if name == "Zhao Jun"  => println("you are a good boy")
      case _ => println("you need to work hard")
    }
  }
}

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

Scala的模式匹配语法,有一个特点在于,可以将模式匹配的默认情况,下划线,替换为一个变量名,此时模式匹配语法就会将要匹配的值赋值给这个变量,从而可以在后面的处理语句中使用要匹配的值
为什么有这种语法??思考一下。因为只要使用用case匹配到的值,是不是我们就知道这个只啦!!在这个case的处理语句中,是不是就直接可以使用写程序时就已知的值!
但是对于下划线这种情况,所有不满足前面的case的值,都会进入这种默认情况进行处理,此时如果我们在处理语句中需要拿到具体的值进行处理呢?那就需要使用这种在模式匹配中进行变量赋值的语法!!

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    judgeScore('F',"Zhao Jun")
    judgeScore('F',"Feng Xiangbin")
  }

  def judgeScore(score:Char, name:String)={
    score match {
      case 'A' => println("Excellent")
      case 'B' => println("Great")
      case 'C' => println("Just so so")
      case _grade if name == "Zhao Jun"  => println("you are a good boy, your score is " + _grade)
      case _grade => println("you need to work hard, your score is " + _grade)
    }
  }
}

对类型进行模式匹配

Scala的模式匹配一个强大之处就在于,可以直接匹配类型,而不是值!!!这点是java的switch case绝对做不到的。
理论知识:对类型如何进行匹配?其他语法与匹配值其实是一样的,但是匹配类型的话,就是要用“case 变量: 类型 => 代码”这种语法,而不是匹配值的“case 值 => 代码”这种语法。

def solveException(e:Exception): Unit = {
  e match {
    case e1:IOException => println("IOException")
    case _:Exception => println("UnKnow Exception")
  }
}

对Array和List进行模式匹配

对Array进行模式匹配,分别可以匹配带有指定元素的数组、带有指定个数元素的数组、以某元素打头的数组
对List进行模式匹配,与Array类似,但是需要使用List特有的::操作符
Array模式匹配

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    greeting(Array("Zhao Jun"))
    greeting(Array("Zhao Jun", "Feng Xiangbin"))
    greeting(Array("girl1", "girl2", "girl3"))
    greeting(Array("Feng Xiangbin"))
  }

  def greeting(list:Array[String]): Unit ={
   list match {
     case Array("Zhao Jun") => println("hello Zhao Jun")
     case Array(girl1, girl2, girl3) => println("hi, girls, nice to meet you " + girl1 + ", " + girl2 + ", " + girl3)
     case Array("Zhao Jun", _*) => println("hello, Zhao Jun, please introduce your friends for us")
     case _ => println("hello, who are you")

   }
  }
}

List模式匹配

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    greeting(List("Zhao Jun"))
    greeting(List("Zhao Jun", "Feng Xiangbin"))
    greeting(List("girl1", "girl2", "girl3"))
    greeting(List("Feng Xiangbin"))
  }
  def greeting(list:List[String]): Unit ={
    list match {
      case "Zhao Jun":: Nil => println("hello Zhao Jun")
      case girl1:: girl2:: girl3 => println("hi, girls, nice to meet you " + girl1 + ", " + girl2 + ", " + girl3)
      case "Zhao Jun" :: tail => println("hello, Zhao Jun, please introduce your friends for us")
      case _ => println("hello, who are you")
    }
  }
}

case class与模式匹配

Scala中提供了一种特殊的类,用case class进行声明,中文也可以称作样例类。case class其实有点类似于Java中的JavaBean的概念。即只定义field,并且由Scala编译时自动提供getter和setter方法,但是没有method。
case class的主构造函数接收的参数通常不需要使用var或val修饰,Scala自动就会使用val修饰(但是如果你自己使用var修饰,那么还是会按照var来)
Scala自动为case class定义了伴生对象,也就是object,并且定义了apply()方法,该方法接收主构造函数中相同的参数,并返回case class对象

class People {

}
case class Student(name:String, classRoom:String) extends People {

}
case class Teacher(name:String, subject:String) extends People {

}
object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    var p = new Teacher("Feng Xiangbin", "Computer")
    judgeProple(p)
    var p1 = new Student("Zhao Jun", "科技园")
    judgeProple(p1)
    var p2 = new People
    judgeProple(p2)
  }

  def judgeProple(p:People): Unit = {
    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 " + classRoom)
      case _ => println("please get out of school")
    }
  }
}

Option与模式匹配
Scala有一种特殊的类型,叫做Option。Option有两种值,一种是Some,表示有值,一种是None,表示没有值。
Option通常会用于模式匹配中,用于判断某个变量是有值还是没有值,这比null来的更加简洁明了
Option的用法必须掌握,因为Spark源码中大量地使用了Option,比如Some(a)、None这种语法,因此必须看得懂Option模式匹配,才能够读懂spark源码。

object MatchCaseStudy {
  def main(args: Array[String]): Unit = {
    judegScore("Zhao Jun")
    judegScore("Zhao J")
  }

  def judegScore(name:String): Unit = {
    var map = Map(("Zhao Jun","B"), ("Feng Xiangbin", "A"))
    var grade = map.get(name)
    grade match {
      case Some(grade) => println("your grade is " + grade)
      case None => println("Sorry, your grade information is not in the system")
    }
  }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,794评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,050评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,587评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,861评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,901评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,898评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,832评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,617评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,077评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,349评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,483评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,199评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,824评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,442评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,632评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,474评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,393评论 2 352

推荐阅读更多精彩内容