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()
  }
}

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

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

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