我正在为 Windows 扩展 Python 2.7.5 应用程序。该应用程序使用 SetUnhandledExceptionFilter
安装在发生未处理的 C 异常时调用的 Python 函数:
@ctypes.WINFUNCTYPE(ctypes.wintypes.LONG, PEXCEPTION_POINTERS)
def crashHandler(exceptionInfo):
# Some code that deals with the unhandled C exception...
...
windll.kernel32.SetUnhandledExceptionFilter(crashHandler)
(我将在下面给出 PEXCEPTION_POINTERS
的代码,因为我认为它与这个问题的目的无关。)
在其原始形式中,crashHandler
执行一些日志记录并关闭进程。我观察到它多次处理未处理的异常,因此我非常有信心 crashHandler
已正确安装为 UnhandledExceptionFilter。
我对 crashHandler
做了一些修改,现在想测试它。为此,我的想法是以编程方式引发一个 C 异常,该异常随后应由 crashHandler
处理。我尝试了以下方法:
>>> windll.kernel32.RaiseException(5, 0, 0, None)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
WindowsError: [Error 5] Access is denied
>>> windll.kernel32.RaiseException(6, 0, 0, None)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
WindowsError: [Error 6] The handle is invalid
因此异常立即被 Python 解释器捕获,因此不会传播到我的 crashHandler
。
我如何才能 a) 禁用 Python 的异常处理或 b) 以编程方式引发未被 Python 的异常处理机制捕获并传播到我的 crashHandler
的 C 异常?
这是 PEXCEPTION_POINTERS
的代码:
from ctypes import Structure
import ctypes
EXCEPTION_MAXIMUM_PARAMETERS = 15
class EXCEPTION_RECORD(Structure):
pass
PEXCEPTION_RECORD = ctypes.wintypes.POINTER(EXCEPTION_RECORD)
EXCEPTION_RECORD._fields_ = [
('ExceptionCode', ctypes.wintypes.DWORD),
('ExceptionFlags', ctypes.wintypes.DWORD),
('ExceptionRecord', PEXCEPTION_RECORD),
('ExceptionAddress', ctypes.wintypes.LPVOID),
('NumberParameters', ctypes.wintypes.DWORD),
('ExceptionInformation', ctypes.wintypes.LPVOID * EXCEPTION_MAXIMUM_PARAMETERS),
]
class EXCEPTION_POINTERS(Structure):
_fields_ = [('ExceptionRecord', PEXCEPTION_RECORD),
('ContextRecord', ctypes.wintypes.LPVOID)]
PEXCEPTION_POINTERS = ctypes.wintypes.POINTER(EXCEPTION_POINTERS)
最佳答案
windll模块自动捕获调用中出现的windows SEH异常,并将其转换为WindowsError python异常。因此它们不是未处理的异常。
您需要使用没有这么好的包装器的 API。
最好的办法是使用 C 扩展模块,它会通过取消引用随机指针(或空指针)来导致访问冲突。
关于python - 在 Python 中引发未处理的 C 异常,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/19154217/