python的try闭包作用域的问题

1,504 阅读2分钟

0x1:

python的try, except 很给力, 在有一些需要捕捉异常的场景下,能够安装异常的类别来进行特定的操作,假如说现在我们有一个场景是“无论产生什么异常,我们都有一个总的捕捉逻辑,这种情况下,我们就要注意try的闭包作用域问题了

0x2,直接上码:


def allCatch(func):
  def wrapper(*args, **kwargs):
    try:
      ret = func(*args, **kwargs)
      return ret
    except Exception as e:
      print "[allcatch] ------------------- ", e.message

  return wrapper



def foo():
  l = [0, 1, 2]
  li = l[9]
  print li


@allCatch
def main():
  foo()


if __name__ == "__main__":
  main()

上面的这个代码,在main里面执行foo能捕捉到总的异常,但是我们现实中的逻辑可能会是下面这样


def allCatch(func):
  def wrapper(*args, **kwargs):
    try:
      ret = func(*args, **kwargs)
      return ret
    except Exception as e:
      print "[allcatch] ------------------- ", e.message

  return wrapper



def foo():
  l = [0, 1, 2]
  li = l[9]
  print li

def foo2():
  try:
    foo()
  except:
    print "foo2 except"

@allCatch
def main():
  foo2()


if __name__ == "__main__":
  main()


上面的代码运行的结果就是

foo2 except

如我们所见, 如果在main里面有另外的函数也执行了try捕捉异常,那么main本身的try是无效的,也就达不到我们要的目的了。所以我们要注意try本身的闭包作用域的问题

0x3 建议的使用姿势

  • 我们的核心流程函数(也就是我们想捕捉的主要流程所在的函数),我们就在这个函数上进行异常捕捉就行,比如上面的例子,其实我们可以改成

@allCatch
def foo():
  l = [0, 1, 2]
  li = l[9]
  print li


def foo2():
  foo()

def main():
  foo2()


if __name__ == "__main__":
  main()

因为我们的主逻辑是在 foo()函数, 所以比较建议的做法就是把我们的异常捕捉放在这些我们需要关注,有可能产生异常的函数主体上,那么问题又来了,假如我有好几个这样的主函数怎么办?那就全部加上。。。。:),所以就要封装好我们的代码,不要没事写那么多垃圾代码才是王道。