Scala模式匹配&样例类&偏函数

模式匹配

  • match不是语句,而是一个表达式,拥有返回值。在match表达式中可以使用任何类型

    val sign = ch match {
      case '+' => 1
      case '*' => 2
      case _ => 0
    }
    
    // 给模式添加守卫,只匹配数字
    val sign = ch match {
      case _ if Character.isDigit(ch) => Character.digit(ch, 10)
    }
    
    // 匹配数组
    val arr = Array(1, 2, 3, 4, 5, 6)
    val result = arr match {
      case Array(1, 2, 3, 4, 5, _) => 1
      case Array(1, 2, _*) => 2
      case _ => 0
    }
    注: _ 表示任意一个元素,_* 表示任意长度的元素
    
    // 类型匹配
    def getType(obj: Any): Unit = {
      obj match {
        case _: Array[String] => println("Array[String]")
        case _: String => println("String")
        case _: Int => println("Int")
        case _: BigInt => println("BigInt")
        case _ => println("unkown")
      }
    }
    
    // 匹配列表
    val result = list match {
      case a :: b :: tail => a + " " + b
      case a :: tail => a
      case _ => "none"
    }
    
    // 匹配元组
    val result = tuple match {
      case (a, _) => a
      case _ => "none"
    }
    注:匹配元组是无法使用 _*
    

    注意:泛型的类型匹配要注意如List[String]、Map[Char,Int]等不会成功匹配,如List[Int]等亦可匹配,因而往往使用通配符List[ _ ]进行匹配,但Array[Int]是可行的

样例类

  • 样例类是一种特殊的类,他们经过优化以被用于模式匹配。
abstract class Solution
case class Add(a: Int, b: Int) extends Solution
case class Mul(a: Int, b: Int) extends Solution
// 还可以有单例的样例对象
case object Nothing extends Solution

def getType(obj: Solution): Any = {
  obj match {
    case Add(a, b) => a + b
    case Mul(a, b) => a * b
    case Nothing => "Nothing"
    case _ => "Null"
  }
}

// 用法
val add = Add(1, 1)
val mul = Mul(10, 10)
val result = getType(add)  // 2
val result1 = getType(mul) // 100

// 可以把模式匹配直接放到公共超类中
abstract class Solution{
  def getType: Any = {
    this match {
      case Add(a, b) => a + b
      case Mul(a, b) => a * b
      case Nothing => "Nothing"
      case _ => "none"
    }
  }
}

// 用法
add.getType
mul.getType

注意: 样例类的实例使用(), 样例对象不使用()。

当声明一个样例类时会自动发生下面几件事:
  • 构造器中的么一个参数都会成为val字段,除非显式的设为var
  • 在伴生对象中提供apply方法,可以不用关键字new就能构造相应对象
  • 提供unapply方法让模式匹配可以工作
  • 生成toString,equals, hashCode,copy方法
密封类
  • 模式匹配完成后需要确保所有的情况都被考虑,要达到这个目的,需要将样例类的公共超类声明为sealed
sealed abstract class Solution
case class Add(a: Int, b: Int) extends Solution
case class Mul(a: Int, b: Int) extends Solution
  • 密封类的所有子类必须都在该密封类所在的文件中定义。
  • 如果一个类是密封的,那么在编译期所有的子类就是可知的,因而编译器可以检查模式语句的完整性。
模拟枚举
  • 在Scala中,样例类可以模拟枚举类型

    sealed abstract class Color
    // 使用单例样例对象
    case object Red extends Color
    case object Blue extends Color
    case object Green extends Color
    
    def getColor(obj: Color): String = {
      obj match {
        case Red => "Red"
        case Blue => "Blue"
        case Green => "Green"
      }
    }
    
    // 用法
    getColor(Red)
    或
    val color = Red
    getColor(color)
    

偏函数

  • 被包在花括号内的一组case语句是一个偏函数——一个并非对所有输入值都有定义的函数。它是PartialFunction[A, B]类的一个实例。(A是参数类型, B是返回类型)。
  • PartialFunction[A, B] 类有两个方法:apply方法从匹配到的模式计算函数值,而isDefinedAt方法在输入至少匹配其中一个模式时返回true
  • 如果把函数用到其不支持的值时会报错
// 定义
val fun: PartialFunction[Int, String] = { case 1 => "yes"; case 0 => "no" }

// 用法
fun(1)          // yes
fun(0)          // no
fun.isDefineAt(1)   // true
fun.isDefineAt(3)   // false
fun(3)          // 报错
  • GenTraversable特质的collect方法将一个偏函数应用到所有在该偏函数有定义的元素,并返回包含这些结果的序列
"1+2*3-4" collect { case '+' => 1; case '*' => 2; case '-' => 1 }   // Vector(1, 2, 1)
  • Actor中的例子
react {
  case (name: String, actor: Actor) => {
    actor ! getip(name)
    act()
  }
  case msg => {
    println("UnHandler message" + msg)
    act()
  }
}

注意:偏函数表达式必须位于编译器可以推断出返回类型的上下文中。把它赋值给一个带有类型声明的变量,或者将它作为参数传递都是可以的。

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

推荐阅读更多精彩内容

  • Scala与Java的关系 Scala与Java的关系是非常紧密的!! 因为Scala是基于Java虚拟机,也就是...
    灯火gg阅读 3,437评论 1 24
  • 1.1. 什么是Scala Scala是一种多范式的编程语言,其设计的初衷是要集成面向对象编程和函数式编程的各种特...
    文子轩阅读 1,531评论 1 3
  • 她为他研过露煮过茶,下水取锦囊,在年家为他毫不犹豫扑向那些刀剑。 ...... 还有许多,如此之多 璇玑笑了笑,“...
    糊凃虫虫阅读 762评论 0 0
  • 今天早上我起的有点晚,做饭的过程中,突然想起老师说今天要打扫卫生,得早点到校。我就赶快让孩子她爸,叫孩子快点起床,...
    shmily123034阅读 96评论 0 0
  • 我孤独前行 在一面镜子面前赤裸身体 带着黑色礼帽 我很高兴我有一双好看的脚 我的房间陈设着简陋的桌椅和电脑 唯一奢...
    泰闲阅读 446评论 8 10