Python的for和break循环结构中使用else语句的技巧

时间:2022-09-23 19:19:07

Python中的while或者for循环之后还可以有else子句,作用是for循环中if条件一直不满足,则最后就执行else语句。

?
1
2
3
4
5
6
7
8
9
10
11
for i in range(5):
 if i == 1:
  print 'in for'
else:
 print 'in else'
 
print 'after for-loop'
 
# in for
# in else
# after for-loop

但我们发现if条件在循环的过程中成立了,最终还是执行了else语句里的内容,这是为什么呢?

好的,我们这时看下面这段程序:

?
1
2
3
4
5
6
7
8
9
10
11
12
for i in range(5):
 if i == 1:
  print 'in for'
  break
else:
 print 'in else'
 
print 'after for-loop'
 
 
# in for
# after for-loop

我们在if里添加了一个break,这是因为else是在for后执行的,但只有for循环正常退出时才会执行else语句(不是由break结束循环)。而当循环是由break语句中断时,else就不被执行。

for/else等效于下面这段代码,可以类似C语言那样添加一个flag:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
found = False
for i in range(5):
 if i == 1:
  found = True
  print 'in for'
if not found:
 print 'not found'
 
print 'after for-loop'
 
 
# in for
# after for-loop

与for语句相似,while语句中的else子句用法是一样的,else块在循环正常结束和循环条件不成立时执行。

对于条件语句 if- else 我们已经很熟悉了, 但是在Python中,for-else用于处理遍历失败。

比如我们要实现这样一个功能:找出(81,99)中最大的完全平方数并输出,找不到则输出提示。

如果用c++的for循环实现,必须手动的判断for循环是否遍历失败:

?
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
#include <iostream>
#include<math.h>
using namespace std;
int main()
{
 int i;
 float n;
 for(i=99;i>81;i--)
 {
  n=sqrt((float)i);
  if(n==int(n))
  {
   cout<<i;
   break;
  }
 }
 if(i==81) //边界判断
  cout<<"didn't find it!"<<endl;
 return 0;
}

而用Python的for-else则可简单的实现这一功能:

?
1
2
3
4
5
6
7
8
from math import sqrt
for n in range(99,81,-1):
 root = sqrt(n)
 if root == int(root):
  print n
  break
else:
 print"Didn't find it!"

在for循环完整完成后才执行else;如果中途从break跳出,则连else一起跳出。

特别需要注意的是如果for中有if语句,else的缩进一定要和for对齐,如果和if对齐,则变成if-else语句,会产生意想不到的错误如下:

?
1
2
3
4
5
6
7
8
from math import sqrt
for n in range(99,81,-1):
 root = sqrt(n)
 if root == int(root):
  print n
  break
 else:
  print"Didn't find it!"

虽然使用for-else节省两行代码同时便于阅读,但是容易和if-else混淆。貌似实际中不会常用,反而更倾向于手动处理。