for循环本质是一种遍历,遍历集合中的每个元素,当遍历完最后一个元素的时候,自动终止.
遍历字符串
val a = "abc" // 字符串
// 遍历容器. 变量c是常量, 不能修改
for (c <- a) {
println(c)
}
结果
a
b
c
字符相加就变成int类型了
val a = "abc" // 字符串
// 遍历容器. 变量c是常量, 不能修改
for (c <- a) {
println(c + 0) //字符相加就变成了int类型了
}
97
98
99
遍历数字
基本遍历
三种写法
// 使用for输出 1-10
// 1.to(10) 是获取到1~10的容器,简写是 1 to 10
for (i <- 1.to(10)) {
println(i)
}
//大于等于1,小于11
for (i <- 1 until 11) {
println(i)
}
for (i <- 1 to 10) {
println(i)
}
得到的结果就是循环遍历1~10
1
2
3
4
5
6
7
8
9
10
获得奇数数字
// 使用for输出 1-10的所有奇数
for (i <- 1 to(10, 2)) { // to有两个参数: 参数1就是结束的值, 参数2是步长
println(i)
}
//上面的简写
for(i <- 1 to 10 by 2 ){
println(i)
}
输出
1
3
5
7
9
从10到1倒着输出
// 使用for输出 10-1
for (i <- 1 to 10 reverse) {
println(i)
}
另外一种写法
for (i <- 10 to 1 by -1) {
println(i)
}
输出:
10
9
8
7
6
5
4
3
2
1
为true就进入循环体
保护式为true则进入循环体内部,为false则跳过,类似于continue。
输出1-10所有的奇数,如果i % 2 == 1 成立的话就输出
for (i <- 1 to 10 if i % 2 == 1) {
if (i % 2 == 1) {
println(i)
}
}
另外一种写法,更简洁,但是我感觉可读性似乎变差.
for (i <- 1 to 10 if i % 2 == 1) {
println(i)
}
1
3
5
7
9
关于代码的精简和可读性
输出1~100 的偶数 , j 是 两个i 相乘 ,只有j小于10000的时候输出j,如果j大于10000就不输出j
下面两个代码片段结果是一个结果,但是一个精简了,但是可读性差了,一个不精简,但是可读性好…
for (i <- 1 to 100) {
if (i % 2 == 0) {
var j = i * i
if (j < 10000) {
println(s"i =$i , j = $j")
}
}
}
for (i <- 1 to 100 if i % 2 == 0; j = i * i if j < 10000) {
println(s"i =$i , j = $j")
}
退出循环
Scala内置控制结构特地去掉了break和continue,是为了更好的适应函数式编程,推荐使用函数式的风格解决break和continue的功能,而不是一个关键字。scala中使用breakable控制结构来实现break和continue功能。
不过这种方式性能不是特别好,点开看源码是自己抛出异常自己捕获的.
案例
break
循环遍历10以内的所有数据,数值为5,结束循环(break)
import scala.util.control.Breaks
object For3 {
def main(args: Array[String]): Unit = {
var n = 1
Breaks.breakable {
while (n < 10) {
println("n=" + n)
n += 1
if (n == 5) {
Breaks.break()
}
}
}
println("exit")
}
}
输出
n=1
n=2
n=3
n=4
exit
continue
循环遍历10以内的所有数据,奇数打印,偶数跳过(continue)
import scala.util.control.Breaks
object For3 {
def main(args: Array[String]): Unit = {
var n = 0
while (n < 10) {
Breaks.breakable {
n += 1
if (n % 2 != 0) {
println(n)
} else {
println("continue")
Breaks.break()
}
}
}
}
}
输出
1
continue
3
continue
5
continue
7
continue
9
continue
注意事项
(1)break:breakable放在循环外
(2)continue:breakable放在循环内
嵌套循环
(1)将一个循环放在另一个循环体内,就形成了嵌套循环。其中,for,while,do…while均可以作为外层循环和内层循环。【建议一般使用两层,最多不要超过3层】
(2)实质上,嵌套循环就是把内层循环当成外层循环的循环体。当只有内层循环的循环条件为false时,才会完全跳出内层循环,才可结束外层的当次循环,开始下一次的循环。
(3)设外层循环次数为m次,内层为n次,则内层循环体实际上需要执行m*n次。
下面两个代码片段是一个作用,还是建议用第二种
第一种
for (i <- 1 to 9; j <- 1 to i) {
print(s"$j * $i = ${i * j}\t")
if (i == j) println()
}
}
第二种
for (i <- 1 to 9) {
for (j <- 1 to i) {
print(s"$j * $i = ${i * j}\t")
}
println()
}
输出:
1 * 1 = 1
1 * 2 = 2 2 * 2 = 4
1 * 3 = 3 2 * 3 = 6 3 * 3 = 9
1 * 4 = 4 2 * 4 = 8 3 * 4 = 12 4 * 4 = 16
1 * 5 = 5 2 * 5 = 10 3 * 5 = 15 4 * 5 = 20 5 * 5 = 25
1 * 6 = 6 2 * 6 = 12 3 * 6 = 18 4 * 6 = 24 5 * 6 = 30 6 * 6 = 36
1 * 7 = 7 2 * 7 = 14 3 * 7 = 21 4 * 7 = 28 5 * 7 = 35 6 * 7 = 42 7 * 7 = 49
1 * 8 = 8 2 * 8 = 16 3 * 8 = 24 4 * 8 = 32 5 * 8 = 40 6 * 8 = 48 7 * 8 = 56 8 * 8 = 64
1 * 9 = 9 2 * 9 = 18 3 * 9 = 27 4 * 9 = 36 5 * 9 = 45 6 * 9 = 54 7 * 9 = 63 8 * 9 = 72 9 * 9 = 81
循环返回值 yield
基本语法
val res = for(i <- 1 to 10) yield i
println(res)
说明:将遍历过程中处理的结果返回到一个新Vector集合中,使用yield关键字,如果循环里面是多行代码的话,那么就返回最后一行代码的值
案例: 将原数据中所有值乘以2,并把数据返回到一个新的集合中。
var res = for (i <- 1 to 10) yield {
i * 2
}
println(res)
输出结果
Vector(2, 4, 6, 8, 10, 12, 14, 16, 18, 20)