《快学Scala》——控制结构和函数
程序员文章站
2022-06-14 22:14:37
...
条件表达式
- 在Scala中if/else表达式有值,这个值就是跟在if或else之后的表达式的值。例如:
if (x > 0) 1 else -1
上述表达式的值是1或-1,具体是哪一个取决于x的值。你可以将if/else的值赋值给变量:
val s = if (x > 0) 1 else -1
这与如下语句的效果一致:
if (x > 0) s = 1 else s = -1
以上两个的区别在于:第一个可以用来初始化val;第二种s必须是var。
- 在Scala中每个表达式都有一个类型。如:
if (x > 0) 1 else -1
的类型是Int,因为两个分支的类型是Int。当两个分支的类型不同时,那该表达式的类型就是两个分支类型的公共超类型。
注:Scala里,每个类都继承自通用的名为Any的超类。因为所有的类都是Any的子类,所以定义在Any中的方法就是“共同的”方法:它们可以被任何对象调用。Scala还在层级的底端定义了一些类,如Null和Nothing,扮演通用的子类。即,Any是所有其他类的超类,Nothing是所有其他类的子类。
- 如果else部分缺失,比如
if (x > 0) 1
,那么该表达式可能没有输出值。但是在Scala中,每个表达式都应该有值。所以在Scala中引入了Unit类,写做()。那么上面的表达式就等同于if (x > 0) 1 else ()
。
块表达式和赋值
- 在Scala中,{}块包含一系列表达式,其结果也是一个表达式。块的最后一个表达式的值就是块的值。如下:
val z = {val x = 2; val y = 4; x * y} //结果z = x * y = 8
- 在Scala中,赋值动作本身是没有值的,或者更严格的说他们的值是Unit类型的。如
{r = r * n; n -= 1}
的值就是Unit类型的。
注:由于赋值语句的值是Unit类型的,故不能将它们串联起来。如x = y = 1
,这里y = 1
的值是()
。
输入和输出
- 输出:用print和println这和Java类似。
- 输入:可以用readLine函数读取控制台上的一行输入。也可以用readInt、readDouble、readByte、readShort、readLong、readFloat、readBoolean或者readChar。和其他方法的区别是,readLine带有一个参数作为提示字符串。如下:
val name = readLine("Your name:")
print("Your age:")
val age = readInt()
循环
Scala中的使用循环有两种选择:
- while:
while (n > 0) {
r = r * n
n -= 1
}
- for:
for(i <- 表达式)
for(i <- 1 to n)
r = r * i
注:在for循环的变量之前并没有val和var的指定。该变量的类型是集合的元素类型。循环变量的作用域一直持续到循环结束。
- 当for循环在遍历字符串和数组时,需要使用until方法而不是to方法。to方法返回包含上限的闭区间,until方法返回一个不包含上限的区间。如下遍历字符串的例子:
//方法一
val s = "Hello"
var sum = 0
for(i <- 0 until s.length)
sum +=s(i)
//方法二
var sum = 0
for(ch <- "Hello") sum += ch
注:Scala不提供break或continue来退出循环,退出循环的方法如下:
- 使用Boolean型的控制变量。
- 使用嵌套函数,从函数中return。
- 使用Breaks对象的break方法。
高级for循环和for推导式
- 可以使用变量<-表达式的形式提供多个生成器,用分号将它们隔开。例如:
for(i <- 1 to 3; j <- 1 to 3) print((10 * i + j) + " ")
//将打印11 12 13 21 22 23 31 32 33
- 每个生成器都可以带一个守卫,以if开头的Boolean表达式。注意if前没有分号例如:
for(i <- 1 to 3; j <- 1 to 3 if i != j) print((10 * i + j) + " ")
//将打印12 13 21 23 31 32
- 如果for循环的循环体使用yield开始,则该循环会构造出一个集合,每次迭代生成集合中的一个值。这类循环成为for推导式。例如:
for(i <- 1 to 10) yield i % 3
//生成 Vector(1, 2, 0, 1, 2, 0, 1, 2, 0, 1)
其中for推导式生成的集合与它的第一个生成器是类型兼容的。如下图:
函数
- Scala除了方法外还支持函数。方法对对象进行操作,函数不是。要定义函数,必须给出函数名、参数和函数体。如下:
def abs(x: Double) = if(x > 0) x else -x
必须给定所有参数的类型,只要函数不是递归的,就不用指定返回类型。如果函数体需要多个表达式,可以使用代码块,块中最后一个表达式的值就是函数的返回值。
默认参数和带名参数
def decorate(str: String, left: String = "[", right:String = "]") = left + str + right
//调用函数如下
decorate("Hello") //输出 [Hello]
decorate("Hello", "%%%") //输出 %%%Hello]
decorate(left = "[", str = "Hello", right = "]") //输出 [Hello]
decorate("Hello",right = "****[") //输出 [Hello***]
decorate("<<","Hello",right = ">>>") //输出 Hello<<>>>
从上例可以看出left和right是默认参数,如果不喜欢默认值就可以在调用的时候重新赋值。我们可以在提供参数值时带上参数名,这个时候参数列表的顺序可以变。当我们混用未命名参数和带名参数时,需要将未命名参数排在前面,并将未命名参数按照参数列表的顺序排放。
过程
Scala中如果函数体包含在花括号当中但没有前面的=
号,那么返回类型应该是Unit。这样的函数被称作过程。
懒值
当val被声明为lazy时,它的初始化将被推迟,直到首次使用时才取值。