Scala控制结构

走过红尘的纷扰,弹落灵魂沾染的尘埃,携一抹淡淡的情怀,迎着清馨的微风,坐在岁月的源头,看时光婆娑的舞步,让自己安静在时间的沙漏里,感受淡如清风,静若兰的唯美。

本章讲述Scala基本的控制结构,包括条件分支,循环;最后讲述自定义控制结构的实现技术,包括条件分支,循环,中止。

  1. 内置控制结构
  • 条件分支
  • 循环结构
  1. 自定义控制结构
  • 条件分支
  • 循环结构
  • 异常终止

内置控制结构

scala中,一切皆为表达式。包括内置的控制结构,包括条件分支,循环结构,for推导式,模式匹配,及其函数调用等。

条件分支

if-else并非语句,而是一个表达式。它拥有一个值,要么来自if分支,要么来自else分支。

def abs(x: Int): Int = if (x > 0) x else -x

根据if (x > 0) x else -x表达式的类型,可以自动推演出abs返回值类型为Int

最近公共父类

两个分支的类型可以不同,将自动推演为两者之间「最近的公共父类」。例如,abs的返回值类型被推演为NoneSome的最近公共父类Option[Int]

def abs(n: Int) = if (n > 0) Some(n) else None

单分支的值

对于单分支if,编译器默认地引入else ()分支。

def abs(n: Int) = if (n > 0) n

等价于

def abs(n: Int) = if (n > 0) n else ()

其中,()的类型为Unit类型,而UnitInt最近公共类型为AnyVal,所以此时abs的返回值类型推演为AnyVal

else抛出异常

else分支抛出异常时,函数返回值类型推演为if分支的值类型。这是因为throw表达式的值类型为Nothing,它是所有类的子类。例如,abs的返回值类型推演为Int

def abs(n: Int) = 
  if (n > 0) n else throw new IllegalArgumentException("negative")

函数式风格

一般地,要从某一个容器中获取值,必须先进行前置的状态查询。本质上,这是一种契约。但遗憾的是,这个契约并没有对用户行为形成强力的约束力,导致既有的约定常常被打破。

这是一种常见的实现模式,例如调用Option在调用get之前,先调用isEmpty查询是否为空。可以通过封装改善设计,OptiongetOrElse替代原来的if-else实现,不仅改善了表达力,而且消除了重复代码。

其中,OptionsgetOrElse方法类似于如下实现。

sealed trait Option[+A] {
  def isEmpty: Boolean
  def get: A

  def getOrElse[B >: A](default: => B): B =
    if (isEmpty) default else get
}

循环结构

Scala中,while/do-while不是语句,而是表达式,只不过它们返回的类型为Unit。例如,factorial用于求解n(n <= 16)的阶乘,使用while迭代的实现方式如下。

def factorial(n: Int): Long = {
  var r = 1; var m = n
  while (m > 0) {
    r = r * m
    m -= 1
  }
  r
}

for推导式

可以使用for推导式改善程序的表达力。

def factorial(n: Int): Long = {
  var r = 1
  for(i < 1 to n) r = r * i
  r
}

i < 1 to n称为for推导式的「生成器」。for推导式是Scala的重要特性之一,将在后续章节重点介绍。

线性递归

也可以使用「模式匹配」实现factorial的递归算法。

def factorial(n: Int): Long = n match {
  case 0 => 1
  case _ => n * factorial(n - 1)
}

但是,该实现不具有尾递归的优化效果。其中,模式匹配是Scala的重要特性之一,将在后续章节重点介绍。

尾递归

为了得到优化的尾递归实现,可以使用「本地方法」设计迭代结构来实现,并显式地加上@tailrec的注解,保证编译时尾递归的优化能力。

def factorial(n: Int): Long = {
  @annotation.tailrec
  def loop(r: Int, n: Int): Long = n match {
    case 0 => r
    case _ => loop(r * n, n - 1)
  }
  loop(1, n)
}

函数式风格

可以使用foldLeft实现factorial,这种函数式的风格简单优雅。

def factorial(n: Int): Long = 
  (1 to n).foldLeft(1) { _ * _ }

自定义控制结构

Scala是一门具有高度可扩展性的程序设计语言,它可以设计出自定义的控制结构,以便提升代码复用价值,并改善代码的表达力。

条件分支

可以使用Scala设计「修饰的条件分支」,它将判断条件置后,而突出执行的意图。首先定义一个Conditional的隐式类,它支持iff/unless两种类型的后缀修饰符,它们两者刚好形成相反的语义。

implicit class Conditional[+T](left: => T) {
  def iff(right: => Boolean): Option[T] = if (right) Some(left) else None
  def unless(right: => Boolean): Option[T] = if (!right) Some(left) else None
}

假设,在当前作用域内,隐式类Conditional可见,则可以按照如下方式使用iff/unless

n iff n % 2 == 0  // None
{ println(n) } unless n % 2 == 0

循环结构

可以设计一个指令式的until函数,它将循环调用block,直至条件cond为真为止(与while相反)。

@annotation.tailrec
def until(cond: => Boolean)(block: => Unit) {
  if (!cond) {
    block
    until(cond)(block)
  }
}

当调用until时,最后一个柯里化参数列表的调用可以使用大括号代替小括号。例如,

var n = 10
var r = 0
until (n == 0) {
  r += n
  n -= 1
}

事实上,它等价于

(0 to 10).sum

前者为指令式的代码风格,后者为函数式的代码风格。

异常终止

为了倡导函数式的代码风格,Scala语言内核拒绝了指令式的break, continue特性。但借助于Scala强大的可扩展能力,可以模拟实现break, continue指令的实现。

探秘forall

Traversable是一个可以遍历的,元素类型为A的集合。它定义了一个foreach的内部迭代器,其它方法的实现都是基于foreach的抽象,使用for推导式实现算法逻辑的。

例如,forall用于判断集合元素是否全部满足于某个谓词,否则返回false。算法无需遍历整个集合,实现使用了break提前终止迭代过程。

Scala为了提高执行效率,对外提供函数式的编程接口,内部实现常常使用指令式的代码风格。

import scala.util.control.Breaks._

trait Traversable[+A] {
  def foreach[U](f: A => U): Unit
  
  def forall(p: A => Boolean): Boolean = {
    var result = true
    breakable {
      for (x <- this if !p(x)) { 
        result = false
        break 
      }
    }
    result
  }
}

探秘breakable/break

事实上,break的工作机制是通过异常来实现的。breakable定义了一个执行op: => Unit的上下文环境,break通过抛出异常终止当前执行的op操作。

class Breaks {
  private val breakException = new BreakControl 

  def break(): Nothing = throw breakException

  def breakable(op: => Unit): Unit = 
    try { 
      op 
    } catch {
      case ex: BreakControl =>
        if (ex ne breakException) throw ex
    }
}

模拟break/continue

break只是终止了breakable的上下文。如果要模拟指令式的break,可以通过将循环放在breakable之内实现。

breakable {
  for (x <- xs) {
    if (cond) break
  } 
}

如果要模拟指令式的continue,可以将循环放在breakable之外实现,break时只是终止了当前的迭代,而不是整个迭代过程。

但很遗憾,虽然它代表了continue的语义,但这里依旧使用了break的命名。

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

推荐阅读更多精彩内容

  • 版权申明:转载请注明出处。文章来源:http://bigdataer.net/?p=253 排版乱?请移步原文获得...
    bigdataer阅读 143评论 0 0
  • Scala程序控制结构 注意:与if不同的是,while与do while不能用作表达式,也即其返回值为Unit,...
    LuciferTM阅读 293评论 0 0
  • 86.复合 Cases 共享相同代码块的多个switch 分支 分支可以合并, 写在分支后用逗号分开。如果任何模式...
    无沣阅读 1,340评论 1 5
  • 所谓水晶的灵性, 如果迷信、偏执、依赖于水晶,或者期望不劳而获,那都是错误的看法与迷失。命运掌握在自己的手中,最终...
    钰婧Erica阅读 1,404评论 0 0
  • 设计模式(Design pattern)是一套被反复使用、多数人知晓的、经过分类编目的、代码设计经验的总结。GoF...
    sindri的小巢阅读 1,677评论 2 11