【问题标题】:How to return a value and raise an Exception如何返回值并引发异常
【发布时间】:2014-02-10 09:10:30
【问题描述】:

我对这个try/except 声明有两个目标。

  1. 如果没有发生问题,它需要return 值 1,如果发生任何问题,则需要 0。
  2. 需要引发异常并结束脚本。

我的return 值有效。我也有SystemExit() 工作。但是在一起,它们不起作用。

我的 Python 脚本(相关):

except IOError:
    value_to_return = 0
    return value_to_return
    raise SystemExit("FOOBAR")

这样,它会完全忽略raise SystemExit("FOOBAR") 行。如何获得返回值并且仍然是raise SystemExit("FOOBAR")?这对某些人来说可能是初级的,但实际上我在这方面遇到了很多困难。

【问题讨论】:

  • 如果你提出 SystemExit 并且实际上希望你的脚本结束,返回值应该去哪里(脚本毕竟结束了)?还是先捕获异常,返回,做一些错误处理,然后退出?
  • 如果你回来,加薪就不会发生。无论如何,你不能两者都做!选择你想要的,退出或返回。
  • 只是猜测,但您可能需要返回 0 并让 调用者 提高 SystemExit。无法在同一个函数中同时执行这两种操作。
  • 您是否要向外壳返回状态码?
  • 我与首席工程师再次核对。他想要的是我的脚本返回一个 0,因为出现了问题,并且脚本应该结束,因为从那时起它就变得无用了。所以我确实需要返回一个 0 并结束脚本。我尝试了以下所有其他答案,均未奏效。

标签: python exception exception-handling return systemexit


【解决方案1】:

我认为您可能正在寻找更像这样的东西:

def some_function():
    # this function should probably do some stuff, then return 1 if
    # it was successful or 0 otherwise.
    pass

def calling_function():
    a = some_function()
    if a == 1:
        raise SystemExit('Get the heck outta here!')
    else:
        # Everything worked!
        pass

【讨论】:

    【解决方案2】:

    归与养是互斥的。

    提高SystemExit 将结束脚本。一些清理例程开始运行,如果调用者真的非常想要,他们可以捕获SystemExit 并取消它,但大多数情况下,您可以将其视为在此处停止执行。调用者永远不会有机会看到返回值或用它做任何有意义的事情。

    返回意味着您希望脚本继续。继续可能意味着让调用者提出SystemExit,或者它可能意味着忽略错误,或者它可能意味着其他东西。这意味着什么取决于您,因为您是编写代码的人。

    最后,你确定你应该处理这个错误吗?仅捕获异常以将其转换为系统关闭可能不是最有用的行为。这不是一种用户友好的问题处理方式,而且它隐藏了您从堆栈跟踪中获得的所有有用的调试信息。

    【讨论】:

    • @ingyhere:不,教程部分很烂。 IIRC 有人从 Github 评论中复制了一个写得很糟糕的描述,而没有仔细检查它。 finally 将暂停异常的传播,然后return 将取消异常并返回"value"
    • language reference 有更准确的描述,特别是“如果 finally 子句执行 return 或 break 语句,则丢弃保存的异常”。语言参考往往比教程好(但语言参考也不完美;例如,它经常没有提到如何通过魔术方法自定义各种操作)。
    • 好的,@user2357112supportsMonica。我运行了一个示例程序,您确实是正确的。 Python.org link I presented above 不幸地逐字声明,“如果异常没有由 except 子句处理,则在 finally 子句执行后重新引发异常。”它还讨论了一个执行 finally 然后异常的示例。不过,这不一定与您写的内容相矛盾。不幸的是,在其他 SO 帖子中存在矛盾。为清楚起见,删除了之前的评论。
    【解决方案3】:

    要退出脚本并返回退出状态,请使用sys.exit()

    import sys
    sys.exit(value_to_return)
    

    【讨论】:

    【解决方案4】:

    你不能同时“raise”和“return”,所以你必须在返回值中添加一个特殊的变量(例如:在元组中)以防出错。

    例如: 我有一个函数(名为“func”),它计算一些东西,即使在计算过程中发生异常,我也需要(部分)结果。在我的示例中,我将使用 KeyboardInterrupt 异常(用户按下 CTRL-C)。

    函数中没有异常处理(这是错误的,万一出现异常函数不返回任何东西):

    def func():
        s=0
        for i in range(10):
            s=s+1
            time.sleep(0.1)
        return s
    
    
    x=0
    try:
        for i in range(10):
           s=func()
           x=x+s
    except KeyboardInterrupt:
        print(x)
    else:
        print(x)
    

    现在我引入一个布尔返回值(在元组中,在原始返回值旁边)来指示是否发生了异常。因为在函数中我只处理了 KeyboardInterrupt 异常,我可以肯定这已经发生了,所以我可以在调用函数的地方提出同样的问题:

    def func():
        try:
            s=0
            for i in range(10):
                s=s+1
                time.sleep(0.1)
        except KeyboardInterrupt: # <- the trick is here
            return s, True        # <- the trick is here
        return s, False           # <- the trick is here
    
    
    x=0
    try:
        for i in range(10):
            s,e=func()
            x=x+s
            if e:                         # <- and here
                raise KeyboardInterrupt   # <- and here
    except KeyboardInterrupt:
        print(x)
    else:
        print(x)
    

    注意:我的例子是 python3。使用了 time 模块(在上面的两个代码中),但我没有导入它只是为了让它更短。如果你想真正尝试一下,请放在开头:

    import time
    

    【讨论】:

      【解决方案5】:

      您可以在调用后使用“returning_value”参数引发错误。

      对您的问题的另一个 Pythonic 答案可能是使用 raise 中的错误参数,然后在您的调用中管理错误以获取值,将其从字符串转换并获取您的“return-ish”。

      def your_f():
          try:
            some_io_thingy_ok()
            return 1
          except IOError:
              raise SystemExit("FOOBAR", 0)
      
      try:
          my_returning_value = your_f()
      except SystemExit as err:
          my_returning_value = err.args[1]
      
      
      print(my_returning_value)
      

      来自Python 3 docs

      当异常发生时,它可能有一个关联的值,也称为 作为异常的论点。论据的存在和类型 取决于异常类型。

      except 子句可以在异常名称之后指定一个变量。这 变量绑定到一个异常实例,并存储了参数 在 instance.args 中。为方便起见,异常实例定义 str(),因此可以直接打印参数,而无需引用 .args。也可以先实例化一个异常 提升它并根据需要为其添加任何属性。

      【讨论】:

        【解决方案6】:

        我在不使用 try 的情况下寻找答案,像这样使用“finally”关键字。如果有人知道,请填写我 这是您问题的答案

        try:
            9/0#sample error "don't know how to make IOError"
        except ZeroDivisionError:
            value_to_return = 0
            raise SystemExit("FOOBAR")
        finally:return value_to_return
        

        【讨论】:

          猜你喜欢
          • 2010-11-06
          • 2013-03-30
          • 2012-10-16
          • 2014-05-16
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2019-04-25
          • 1970-01-01
          相关资源
          最近更新 更多