V2EX = way to explore
V2EX 是一个关于分享和探索的地方
现在注册
已注册用户请  登录
推荐学习书目
Learn Python the Hard Way
Python Sites
PyPI - Python Package Index
http://diveintopython.org/toc/index.html
Pocoo
值得关注的项目
PyPy
Celery
Jinja2
Read the Docs
gevent
pyenv
virtualenv
Stackless Python
Beautiful Soup
结巴中文分词
Green Unicorn
Sentry
Shovel
Pyflakes
pytest
Python 编程
pep8 Checker
Styles
PEP 8
Google Python Style Guide
Code Style from The Hitchhiker's Guide
gap
V2EX  ›  Python

麻烦帮忙看看这道简单的题,为什么 finally 会被执行两次呢?

  •  
  •   gap · 2017-07-08 10:06:27 +08:00 · 2694 次点击
    这是一个创建于 2481 天前的主题,其中的信息可能已经有所发展或是发生改变。
    def fancy_divide(numbers, index):
        try:
            denom = numbers[index]
            for i in range(len(numbers)):
                numbers[i] /= denom
        except IndexError:
            fancy_divide(numbers, len(numbers) - 1)
        except ZeroDivisionError:
            print("-2")
        else:
            print("1")
        finally:
            print("0")
    
    fancy_divide([0, 2, 4], 4)
    

    结果是1 0 01 0我理解,但为什么最后还有个 0 呢?finally被执行两次?

    8 条回复    2017-07-08 11:37:51 +08:00
    Herobs
        1
    Herobs  
       2017-07-08 10:13:53 +08:00 via Android   ❤️ 1
    递归完了才会执行第一次的 finally
    mianju
        2
    mianju  
       2017-07-08 10:15:14 +08:00   ❤️ 1
    你第一次触发了 IndexError,在处理完 IndexError 异常后是会调用 finally,加上 fancy_divide(numbers, len(numbers) - 1)里执行的 finally,总共两次 finally 没问题呀。
    mianju
        3
    mianju  
       2017-07-08 10:15:38 +08:00   ❤️ 1
    其实打个断点就能知道了
    Herobs
        4
    Herobs  
       2017-07-08 10:15:50 +08:00 via Android   ❤️ 1
    或者这么理解,finally 一定会执行,你这个函数调用了两次,那肯定会执行两次。
    zjuhwc
        5
    zjuhwc  
       2017-07-08 10:17:55 +08:00 via iPhone   ❤️ 2
    第一次执行触发 indexerror,然后再递归执行一次,第二次正常执行,打印 else 里的 1,打印 finally 里的 0,然后回到第一次执行的函数里,打印 finally 里的 0。

    finally 的含义是不管函数是否抛出异常都会执行。

    另外,建议这种问题用单步调试来看每一步执行过程,比较容易理解
    icenan2
        6
    icenan2  
       2017-07-08 10:20:31 +08:00   ❤️ 1
    递归啊,1 0 都理解,最后的 0 更好理解了嘛
    gap
        7
    gap  
    OP
       2017-07-08 10:25:13 +08:00
    原来如此,感谢大家!
    suueyoung
        8
    suueyoung  
       2017-07-08 11:37:51 +08:00
    递归里面还是 call 了一遍 finally
    关于   ·   帮助文档   ·   博客   ·   API   ·   FAQ   ·   我们的愿景   ·   实用小工具   ·   1509 人在线   最高记录 6543   ·     Select Language
    创意工作者们的社区
    World is powered by solitude
    VERSION: 3.9.8.5 · 25ms · UTC 17:09 · PVG 01:09 · LAX 10:09 · JFK 13:09
    Developed with CodeLauncher
    ♥ Do have faith in what you're doing.