如何在不知道类型的情况下在 python 中捕获异常并获取对异常的引用?
声明:本页面是StackOverFlow热门问题的中英对照翻译,遵循CC BY-SA 4.0协议,如果您需要使用它,必须同样遵循CC BY-SA许可,注明原文地址和作者信息,同时你必须将它归于原作者(不是我):StackOverFlow
原文地址: http://stackoverflow.com/questions/2045786/
Warning: these are provided under cc-by-sa 4.0 license. You are free to use/share it, But you must attribute it to the original authors (not me):
StackOverFlow
How to catch an exception in python and get a reference to the exception, WITHOUT knowing the type?
提问by TM.
I'm wondering how I can catch anyraised object (i.e. a type that does not extend Exception
), and still get a reference to it.
我想知道如何捕获任何凸起的对象(即不扩展的类型Exception
),并且仍然获得对它的引用。
I came across the desire to do this when using Jython. When calling a Java method, if that method raises an exception, it will not extend Python's Exception
class, so a block like this will not catch it:
我在使用 Jython 时遇到了这样做的愿望。调用 Java 方法时,如果该方法引发异常,则不会扩展 Python 的Exception
类,因此像这样的块不会捕获它:
try:
# some call to a java lib that raises an exception here
except Exception, e:
# will never be entered
I can do this, but then I have no access to the exception object that was raised.
我可以这样做,但是我无法访问引发的异常对象。
try:
# some call to a java lib that raises an exception here
except:
# will enter here, but there's no reference to the exception that was raised
I can solve this by importing the Java exception type and catching it explicitly, but this makes it difficult/impossible to write generic exception handling wrappers/decorators.
我可以通过导入 Java 异常类型并显式捕获它来解决这个问题,但这使得编写通用异常处理包装器/装饰器变得困难/不可能。
Is there a way to catch some arbitrary exception and still get a reference to it in the except
block?
有没有办法捕获一些任意异常并仍然在except
块中获得对它的引用?
I should notethat I'm hoping for the exception handling decorator I am making to be usable with Python projects, not just with Jython projects. I'd like to avoid importing java.lang.Exception
because that just makes it Jython-only. For example, I figure I can do something like this (but I haven't tried it), but I'd like to avoid it if I can.
我应该注意,我希望我正在制作的异常处理装饰器可用于 Python 项目,而不仅仅是 Jython 项目。我想避免导入,java.lang.Exception
因为这只会使它成为 Jython-only。例如,我认为我可以做这样的事情(但我还没有尝试过),但如果可以的话,我想避免它。
try:
# some function that may be running jython and may raise a java exception
except (Exception, java.lang.Exception), e:
# I imagine this would work, but it makes the code jython-only
回答by Brian McKenna
You can reference exceptions using the sys
module. sys.exc_info
is a tuple of the type, the instance and the traceback.
您可以使用该sys
模块引用异常。sys.exc_info
是类型、实例和回溯的元组。
import sys
try:
# some call to a java lib that raises an exception here
except:
instance = sys.exc_info()[1]
回答by Adrian Mehlig
FWIW, I have found that if you add this import to your Jython script:
FWIW,我发现如果您将此导入添加到 Jython 脚本中:
from java.lang import Exception
and just use the conventional Python Exception handler:
只需使用传统的 Python 异常处理程序:
except Exception, e:
it will catch bothPython exceptions andJava exceptions
它会赶上两个Python的异常和Java异常
回答by mike rodent
Just for anyone interested... I spent a bit of time testing stuff because I wanted to find out how to get a proper stack trace whether a Python Exception (BaseException in fact, which is the base class) or a java.lang.Throwable (java base class for Exception, Error, etc.) is thrown... this code illustrates how to catch all line number refs correctly.
只是为了任何感兴趣的人......我花了一些时间测试东西,因为我想找出如何获得正确的堆栈跟踪,无论是 Python 异常(实际上是 BaseException,它是基类)还是 java.lang.Throwable (异常、错误等的 java 基类)被抛出......此代码说明了如何正确捕获所有行号引用。
import sys
import traceback
import java
print "hello world"
def custom_hook( type, exc, tb ):
if isinstance( sys.exc_info()[ 1 ], java.lang.Throwable ):
sys.stderr.write( "AS JAVA:\n" )
sys.exc_info()[ 1 ].printStackTrace() # java part
else:
sys.stderr.write( "NO JAVA TRACE:\n" )
sys.stderr.write( "AS PYTHON:\n" )
traceback.print_exc()
# useful for custom exception handling!
sys.excepthook = custom_hook
def handle_exc():
# generate either a java.lang.Throwable (uncomment the next line and comment out "a = 16 / 0"
# java.lang.String( None )
# OR... a python-style BaseException:
a = 16 / 0
class Task( java.lang.Runnable ):
def run( self ):
# NB the problem with all this stuff is that the Java stack trace shows
# a java.lang.Throwable occurring at the last line of this code block...
# print "lots of stuff first"
# print "lots 2"
# handle_exc()
# print "lots 3"
# print "lots of stuff after"
try:
print "lots of stuff first"
print "lots 2"
handle_exc()
print "lots 3"
print "lots of stuff after"
# NB do not catch both (Python) BaseException and java.lang.Throwable...
# except ( BaseException, java.lang.Throwable ), e:
# the above means that the line no. in handle_exc is not shown when a BaseException
# is thrown...
except java.lang.Throwable, t:
tb = sys.exc_info()[ 2 ]
sys.stderr.write( "java.lang.Throwable thrown at: %s\n" % tb.tb_lineno )
raise t
java.awt.EventQueue.invokeAndWait( Task() )
After this one might think of writing a decorator to precede def run( self ) and similar methods so that you don't have to write out this catch-the-Throwable try-except block each time... specifically:
在此之后,您可能会想到在 def run( self ) 和类似方法之前编写一个装饰器,这样您就不必每次都写出这个 catch-the-Throwable try-except 块……特别是:
def throw_trap( function ):
def wrapper(*args, **kvargs):
try:
return function( *args, **kvargs )
except java.lang.Throwable, t:
tb = sys.exc_info()[ 2 ]
while( tb ):
sys.stderr.write( "thrown at: %s\n" % tb.tb_lineno )
tb = tb.tb_next
raise t
return wrapper
def handle_exc():
java.lang.String( None )
# a = 16 / 0
class Task( java.lang.Runnable ):
@throw_trap
def run( self ):
print "lots of stuff first"
print "lots 2"
handle_exc()
print "lots 3"
print "lots of stuff after"
java.awt.EventQueue.invokeAndWait( Task() )