Python中被忽略的else

else, 我们再熟悉不过了。对于一个Python程序员来说,else往往都是配合 if 来使用的,像这样:

a = '12'
if a == '123':
    print(a)
else:
    print('出错了!')

但是,Python中的else并不只能用在if之后,so,这次我们讨论一下Python流程控制中的else。

else子句不仅能在 if 语句中使用,还能在 for、while 和 try 语句中使用,这个语言特性不是什么秘密,但却没有得到重视。我们看一个例子:

my_list = ['apple', 'pear', 'orange', 'banana']
for item in my_list:
    if item == 'banana':
        print('Founded!')
        break
else:
    raise ValueError('No banana flavor found!')

本例当中,循环最后找到了'banana‘,输出'Founded!',并且跳出循环,所以else字句并没有被执行。但如果,将代码修改一下,去掉列表中的'banana':

my_list = ['apple', 'pear', 'orange']
for item in my_list:
    if item == 'banana':
    print('Founded!')
    break
else:
    raise ValueError('No banana flavor found!')

运行代码就会直接抛出错误!如果不使用else字句来完成上述功能,可能我们就需要设置控制标志了,像这样:

my_list = ['apple', 'pear', 'orange']
flag = True for item in my_list:
    if item == 'banana':
        print('Founded!')
        flag = False
        break
if flag:
    raise ValueError('No banana flavor found!')

很明显,这里使用了额外的变量 flag 和 if 语句。

while 和 for 相类似,简单举个例子:

a = 'apple'
while a == 'banana':
    pass
else:
    raise ValueError('No banana flavor found!')

下面看一下try:

try:
    dangerous_call()
except OSError:
    log('OSError...')
else:
    after_call()

很明确,try 块防守的是 dangerouscall() 可能出现的错误,而且很明显,只有 try 块不抛出异常,才会执行 aftercall()。

现在,总结一下 else 子句的行为如下:

for: 仅当 for 循环运行完毕时(即 for 循环没有被 break 语句中止)才运行 else 块。
while: 仅当 while 循环因为条件为假值而退出时(即 while 循环没有被break 语句中止)才运行 else 块。
try: 仅当 try 块中没有异常抛出时才运行 else 块。

即,如果异常或者 return、break 或 continue 语句导致控制权跳到了复合语句的主块之外,那么else 子句也会被跳过。

for/else、while/else 和 try/else 的语义关系紧密,不过与if/else 差别很大。主要是else 这个单词的意思阻碍了我们对这些特性的理解。 按正常的理解应该是“要么运行这个循环,要么做那件事”。可是,在循环中,else 的语义恰好相反:“运行这个循环,然后做那件事。”不过,相信多使用几次,你会熟悉的。

Related post

微信公众号:程序员到架构师

最新文章

Return Top