对任何编程语言来说,都离不开判断、选择、循环等基本的程序控制结构。自然,Scala也实现了必需的基本控制结构,只不过这些内建控制结构的语法更贴近函数式的风格。本章内容将对这些语法逐一讲解,这些语法在Chisel里编写电路逻辑时也是经常出现的。
一、if表达式
用于判断的“if......else if......else”语法想必是所有编程语言都具备的。Scala的if表达式与大多数语言是一样的。在if和每个else if后面都将接收一个Boolean类型的表达式作为参数,如果表达式的结果为true,就执行对应的操作,否则跳过。每个分支都可以包含一个表达式作为执行体,如果有多个表达式,则应该放进花括号里。对整个if表达式而言,实际是算作一个表达式。例如:
scala> def whichInt(x: Int) = {
| if(x == 0) "Zero"
| else if(x > 0) "Positive Number"
| else "Negative Number"
| }
whichInt: (x: Int)Stringscala> whichInt(-1)
res0: String = Negative Number
二、while循环
Scala的“while”语法与C语言一致,都是当判别式的结果为true时,一直执行花括号里的循环体,直到判别式为false。“do......while”也是一样的,先执行一次循环体,再来进行判别,直到判别式为false。例如要计算两个整数的最大公约数:
def gcdLoop(x: Long, y: Long): Long = {
var a = x
var b = y
while (a != 0) {
val temp = a
a = b % a
b = temp
}
b
}
从上述代码可以看出,while语法的风格是指令式的。实际上,Scala把“if”叫“表达式”,是因为if表达式能返回有用的值,而“while”叫循环,是因为while循环不会返回有用的值,主要作用是不断重写某些var变量,所以while循环的类型是Unit。在纯函数式的语言里,只有表达式,不会存在像while循环这样的语法。Scala兼容两种风格,并引入了while循环,是因为某些时候用while编写的代码可阅读性更强。但其实所有的while循环都可以通过其它函数式风格的语法来实现,常见做法就是函数的自我递归调用。例如,一个函数式风格的求取最大公约数的函数定义如下:
def gcd(x: Long, y: Long): Long =
if (y == 0) x else gcd(y, x % y)
三、for表达式与for循环
要实现循环,在Scala里推荐使用for表达式。不过,Scala的for表达式是函数式风格的,没有引入指令式风格的“for(i = 0; i < N; i++)”。一个Scala的for表达式的一般形式如下:
for( seq ) yield expression
整个for表达式算一个语句。在这里,seq代表一个序列。换句话说,能放进for表达式里的