以独立于平台的方式处理特定于Windows的异常

时间:2009-07-31 00:36:15

标签: python windows exception exception-handling

考虑以下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:
    ...

因为在那些甚至没有在例外模块中定义WindowsError的Unix系统上这不起作用。

有没有一种优雅的方法来处理这个错误?

3 个答案:

答案 0 :(得分:13)

如果您需要使用可能并不总是存在的名称捕获异常,请创建它:

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

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

如果你在Windows上,你将使用真正的WindowsError类并捕获异常。如果不是,您将创建一个永远不会引发的WindowsError类,因此except子句不会导致任何错误,并且永远不会调用except子句。

答案 1 :(得分:4)

这是我当前的解决方案,但我在一个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

答案 2 :(得分: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