0.前言
类型检查和类型转换在每个语言外面都有对应实现,比方Java
中的instanceof
和 isInstance
,当然Scala语言也有,然而绝对于其余语言,Scala
为了简化开发,产生了弱小的模式匹配,其原理和Java
中的switch-case
很相似,然而其匹配能力更强,不仅仅能够匹配值,匹配类型,也能够进行类匹配,还能够进行前缀类匹配,而且在Spark
源码中大量应用了模式匹配,另外的就是隐式转换,在另外一篇文章解说,本文从类型检查和类型转换动手,引入Scala
常见的几种模式匹配及其应用。
1.类型查看
要测试某个对象是否属于某个给定的类,能够用isInstanceOf
办法。如果测试胜利,能够用asInstanceOf
办法进行类
型转换。
if(p.isInstanceOf[Employee]){//s的类型转换为Employeeval s = p.asInstanceOf[Employee]}
类型转换次要有上面几个点要留神的:
- 如果p指向的是
Employee
类及其子类的对象,则p.isInstanceOf[Employee]
将会胜利。 - 如果p是null,则
p.isInstanceOf[Employee]
将返回false,且p.asInstanceOf[Employee]
将返回null。 - 如果p不是一个
Employee
,则p.asInstanceOf[Employee]
将抛出异样。 - 如果想要测试p指向的是一个
Employee
对象但又不是其子类,能够用:if(p.getClass == classOf[Employee])
classOf
办法定义在scala.Predef
对象中,因而会被主动引入。不过,与类型检查和转换相比,模式匹配通常是更好的抉择。
p match{//将s作为Employee解决case s: Employee => ...//p不是Employee的状况case _ => ....}
2.模式匹配
Scala 中的模式匹配总结来说反对上面几种匹配:值匹配,类型匹配,汇合元素,元组匹配,有值或者无值匹配,上面从代码角度来看看这几种匹配如何应用,首先模式匹配的语法结构如下:
变量 match { case xx => 代码}
和Java
的不同点,不须要指定break
,即有break
的成果,应用占位符_
来代表默认值,另外match
和 if
一样有返回值。
2.1 值匹配
值匹配,即相似Java
中的整型,字符或者,字符串匹配。然而其之处守卫式匹配(能够了解为默认状况下的条件匹配)
//字符匹配def main(args: Array[String]): Unit = { val charStr = '6' charStr match { case '+' => println("匹配上了加号") case '-' => println("匹配上了减号") case '*' => println("匹配上了乘号") case '/' => println("匹配上了除号") //留神:不满足以上所有状况,就执行上面的代码 case _ => println("都没有匹配上,我是默认值") }}//字符串匹配def main(args: Array[String]): Unit = { val arr = Array("hadoop", "zookeeper", "spark") val name = arr(Random.nextInt(arr.length)) name match { case "hadoop" => println("大数据分布式存储和计算框架...") case "zookeeper" => println("大数据分布式协调服务框架...") case "spark" => println("大数据分布式内存计算框架...") case _ => println("我不意识你...") }}//守卫式匹配def main(args: Array[String]): Unit = { //守卫式 val character = '*' val num = character match { case '+' => 1 case '-' => 2 case _ if character.equals('*') => 3 case _ => 4 } println(character + " " + num)}
2.2 类型匹配
类型匹配是绝对于Java
来说的劣势点,Java
是做不到的,匹配格局如下:case 变量名:类型
,变量名能够用_
来代替
//类型匹配 def typeMathc (x: Any) = { x match { case _: String => println("字符串") case _: Int => println("整形") case _: Array[Int] => println("正星星数组") case _ => println("nothing") } }
2.3 匹配数组,元组,汇合
不同于类型匹配的是,类型匹配只能匹配到整个大的类型,而这种匹配能够匹配相似像某类,然而又能够限度匹配类的局部元素
//数组模式匹配 def arrayMatch(x: Array[Int]) = { x match { case Array(1,x,y) => println(x + ":" + y) case Array(1) => println("only 1 ....") case Array(1,_*) => println("1 结尾的") case _ => println("nothing....") } } //list模式匹配 def listMatch() = { val list = List(3, -1) //对List列表进行模式匹配,与Array相似,然而须要应用List特有的::操作符 //结构List列表的两个根本单位是Nil和::,Nil示意为一个空列表 //tail返回一个除了第一元素之外的其余元素的列表 //别离匹配:带有指定个数元素的列表、带有指定元素的列表、以某元素结尾的列表 list match { case x :: y :: Nil => println(s"x: $x y: $y") case 0 :: Nil => println("only 0") case 1 :: _ => println("1 ...") case _ => println("something else") } } //元组匹配 def tupleMatch() = { val tuple = (5, 3, 7) tuple match { case (1, x, y) => println(s"1, $x, $y") case (_, z, 5) => println(z) case _ => println("else") } }
当数组内没有写值,上面几种匹配等效,任意参数等于齐全类型匹配
case Array(_*) => println("*")case _: Array[Int] => println("整个数组")
2.4 样例类匹配
case class样例类是Scala中非凡的类。当申明样例类时,以下事件会主动产生:
- 主构造函数接管的参数通常不须要显式应用var或val润饰,Scala会主动应用val润饰
- 主动为样例类定义了伴生对象,并提供apply办法,不必new关键字就可能结构出相应的对象
- 将生成toString、equals、hashCode和copy办法,除非显示的给出这些办法的定义
- 继承了Product和Serializable这两个特质,也就是说样例类可序列化和可利用Product的办法
case class是多例的,前面要跟结构参数,case object是单例的。
此外,case class样例类中能够增加办法和字段,并且可用于模式匹配。怎么了解样例类的模式匹配呢,在应用动静绑定时候,从样例类的继承中能够判断,某个对象是否属于某个子类对象,而面向父类的接口,能够简化编程的设计。跟第一局部说到的isInstanceOf
相似,同时,样例类能够承受输出参数进行对应的子类操作。
class Amount//定义样例类Dollar,继承Amount父类case class Dollar(value: Double) extends Amount//定义样例类Currency,继承Amount父类case class Currency(value: Double, unit: String) extends Amount//定义样例对象Nothing,继承Amount父类case object Nothing extends Amountobject CaseClassDemo { def main(args: Array[String]): Unit = { judgeIdentity(Dollar(10.0)) judgeIdentity(Currency(20.2,"100")) judgeIdentity(Nothing) } //自定义办法,模式匹配判断amt类型 def judgeIdentity(amt: Amount): Unit = { amt match { case Dollar(value) => println(s"$value") case Currency(value, unit) => println(s"Oh noes,I got $unit") case _: Currency => println(s"Oh noes,I go") case Nothing => println("Oh,GOD!") } }}
2.5 有值无值匹配
Scala Option
选项类型用来示意一个值是可选的,有值或无值。Option[T]
是一个类型为 T 的可选值的容器,能够通过get()
函数获取Option
的值。如果值存在,Option[T]
就是一个Some
。如果不存在,Option[T]
就是对象 None
。Option
通常与模式匹配联合应用,用于判断某个变量是有值还是无值,上面以Map
的自带返回值Option
来看看这种匹配。
val grades = Map("jacky" -> 90, "tom" -> 80, "jarry" -> 95) def getGrade(name: String): Unit = { val grade: Option[Int] = grades.get(name) grade match { case Some(grade) => println("问题:" + grade) case None => println("没有此人问题!") } } def main(args: Array[String]): Unit = { getGrade("jacky") getGrade("张三") }
3. 总结
Scala
中模式匹配,是用来简化代码,实现更多弱小的不确定或确定的匹配,同时还能够自定义模式匹配,其匹配时候主动利用了apply
和 unapply
办法。
吴邪,小三爷,混迹于后盾,大数据,人工智能畛域的小菜鸟。
更多请关注