目录
简介
我们知道Scala中循环是没有像java中break、continue关键字结束循环的,cscala中的break()方法和breakable()方法用于实现循环中的break中断循环执行功能,两个方法的代码实现也体现了scala函数式编程的思想。
循环中断具体实现方式
代码如下
/**
* breakable的用法
* 1、首先需要导入scala.util.control.Breaks._这个包
* 2、将需要通过breakable控制的代码放到breakable的大括号中
*/
import scala.util.control.Breaks._
//相当于break,跳出整个循环
breakable{
val flag = true
for (i <- 1 to 10 if flag){
println(i)
if (i == 5){
break
}
}
}
//相当于continue,跳出本次循环,继续执行下一次循环
for(j <- 1 to 10 if flag){
breakable{
if(j==7) {
break()
}
println(j)
}
}
将整个循环包含在breakable()方法中,一般来说,循环实现为多行代码,所以可以使用 {} 代替 () 。
在 if 判断之后,其代码块中直接调用 break() 方法,即可使用跳出循环的效果。
两个方法的源码解释
break()方法源码
/**
* Break from dynamically closest enclosing breakable block using this exact
* `Breaks` instance.
*
* @note This might be different than the statically closest enclosing block!
*/
def break(): Nothing = { throw breakException }
从源码中可以看出,该方法无返回值,并且只有一行很简单的代码,就是抛出 breakException 异常,所以调用该方法,不会对原先的代码执行效果造成任何影响,只会中断代码的执行,毕竟是代码执行出现了异常。
我们肯定会想,出异常程序不是就会结束了不执行报错了吗,我们break方法是不是在breakable()这个方法里面,异常就是上升一个层次到breakable()方法里面,那让我们看看breakable()方法里面是怎样执行的吧
breakable()方法源码
/**
* A block from which one can exit with a `break`. The `break` may be
* executed further down in the call stack provided that it is called on the
* exact same instance of `Breaks`.
*/
def breakable(op: => Unit) {
try {
op
} catch {
case ex: BreakControl =>
if (ex ne breakException) throw ex
}
}
从源码中可以看出,该方法参数要求传入一个无返回值的函数,并且该方法无返回值。
该方法内部通过 try 代码块将函数包起来,并 catch 捕获传入的函数执行时抛出的异常。捕获到异常之后,判断该异常是否为 breakException 类型,如果不是,则抛出捕获到的异常;如果是,则什么都不做。
什么都不做非常重要,因为将 for 循环代码块包含在 breakable() 中作为参数传入 breakable() 方法,for 循环代码块中调用了 break() 方法之后,在符合中断条件执行 break() 方法之后,for 循环执行由于出现了异常,停止继续执行,而 breakable() 方法则会捕获到 break() 方法执行时抛出的 breakException 异常,然后判断到该异常为 breakException 异常,什么都不做,因此 breakable() 方法之后的代码就可以继续执行了。scala 设计的 breakable() 方法和 break() 方法联合使用,达到了 java 中 break 关键字的循环中断效果。
Breaks 单例对象代码
/** An object that can be used for the break control abstraction.
* Example usage:
* {{{
* import Breaks.{break, breakable}
*
* breakable {
* for (...) {
* if (...) break
* }
* }
* }}}
*/
object Breaks extends Breaks
代码文档注释中也写出了实现中断 for 循环的案例。if (...) break 这儿的 break 和 break() 效果一样,只不过 scala 中规定,如果方法没有参数,则调用方法时,后面的小括号可以省略。