以与平台无关的方式处理 Windows 特定异常

9
考虑以下 Python 异常:
  [...]
    f.extractall()
  File "C:\Python26\lib\zipfile.py", line 935, in extractall
    self.extract(zipinfo, path, pwd)
  File "C:\Python26\lib\zipfile.py", line 923, in extract
    return self._extract_member(member, path, pwd)
  File "C:\Python26\lib\zipfile.py", line 957, in _extract_member
    os.makedirs(upperdirs)
  File "C:\Python26\lib\os.py", line 157, in makedirs
    mkdir(name, mode)
WindowsError: [Error 267] The directory name is invalid: 'C:\\HOME\\as\
\pypm-infinitude\\scratch\\b\\slut-0.9.0.zip.work\\slut-0.9\\aux'

我希望处理这个特定的异常 - 即,错误号为267的WindowsError。然而,我不能简单地执行以下操作:

try:
    do()
except WindowsError, e:
    ...

因为在Unix系统上,exceptions模块中甚至没有定义WindowsError,所以那种方法行不通。 有没有一种优雅的方式来处理这个错误?

请参考:http://bugs.python.org/issue6609 - Sridhar Ratnakumar
3个回答

15

如果你需要捕获一个可能不存在的异常名称,那么就创建它:

if not getattr(__builtins__, "WindowsError", None):
    class WindowsError(OSError): pass

try:
    do()
except WindowsError, e:
    print "error"

如果你是在Windows上,你将使用真正的WindowsError类并捕获异常。如果不是,在这种情况下,你将创建一个从未被引发的WindowsError类,以便except子句不会导致任何错误,并且except子句将永远不会被调用。


7

这是我的当前解决方案,但我略微厌恶在 except 块中使用非平凡代码:

        try:
            f.extractall()
        except OSError, e:
            # http://bugs.python.org/issue6609
            if sys.platform.startswith('win'):
                if isinstance(e, WindowsError) and e.winerror == 267:
                    raise InvalidFile, ('uses Windows special name (%s)' % e)
            raise

1

@Glenn Maynard的回答不允许使用pdb进行调试,因为WindowsError内置在调试器中不可用。此代码块将在Python调试器和正常执行期间工作:

import exceptions

if not getattr(exceptions, "WindowsError", None):
        class WindowsError(OSError): pass

这个解决方案也是可行的,避免了字符串字面量和完整异常库的导入:

try:
    from exceptions import WindowsError
except ImportError:
    class WindowsError(OSError): pass

请注意,exceptions 库适用于 Python 2 而不是 Python 3。 - Jason S

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接