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

9 投票
3 回答
9364 浏览
提问于 2025-04-15 13:17

考虑一下下面这个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'

我想处理这个特定的错误,也就是WindowsError,错误编号是267。不过,我不能简单地这样做:

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

因为在Unix系统上,这样做是行不通的,因为WindowsError在异常模块中根本没有定义。

有没有什么好的方法来处理这个错误呢?

3 个回答

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
15

如果你需要处理一个可能不总是存在的错误,那么你可以自己创建一个:

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

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

如果你在Windows系统上,你会使用真正的WindowsError类来捕捉这个错误。如果你不在Windows上,你就可以创建一个WindowsError类,这个类不会被触发,这样在处理错误的部分就不会出现任何问题,也就不会执行到处理错误的代码。

7

这是我现在的解决方案,不过我有点不喜欢在异常处理块中使用复杂的代码:

        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

撰写回答