13

我试图在 python 2.7 中定义我自己的异常类,派生自BaseException.

class NestedCommentException(BaseException):
    """
    Exception for nested comments
    """
    def __init__(self, file_path, list_lines):
        self.file_path = file_path
        self.list_lines = list_lines

    def __repr__(self):
        return self.__str__()

    def __str__(self):
        return 'File {0} contains nested comments at lines {1}'.format(self.file_path, ', '.join(self.list_lines))

但是扔的时候不能打印:raise NestedCommentException(file_path, list_lines)触发器

Traceback (most recent call last):
  File "D:\DATA\FP12210\My Documents\Outils\SVN\05_impl\2_tools\svn_tag_setup.py", line 85, in <module>
    tag_checks()
  File "D:\DATA\FP12210\My Documents\Outils\SVN\05_impl\2_tools\svn_tag_setup.py", line 66, in tag_checks
    check_nested_comments(ddl_path)
  File "D:\DATA\FP12210\My Documents\Outils\SVN\05_impl\2_tools\svn_tag_setup.py", line 54, in check_nested_comments
    raise NestedCommentException(file_path, list_lines)
NestedCommentException: <unprintable NestedCommentException object>

即使我定义了__str____repr__方法,您能否解释一下为什么会发生这种情况?

4

2 回答 2

12

TL;博士

__str__()当你看到这个东西时,它基本上意味着你的对象已经引发了某种异常。因此,除非问题微不足道,一眼就能看出来(例如被遗忘的“%s”),否则

  • 将主体包裹__str__在 try/except 子句中作为 Anurag 建议,或

  • 实例化您的异常并__str__在回溯模块之外手动调用(或您可能已覆盖的任何方法),以便您获得异常的完整描述。

分析

实际上,这<unprintable MyException object>可以来自 traceback 模块中的各种函数,当尝试获取值(异常)的字符串(即“可打印”)版本时,它

  1. 调用str()它,如果出现任何问题,

  2. 尝试将其视为 unicode 并将其转换为 ASCII,如果仍然出现任何问题

  3. 只需打印上述表示。

负责代码(2.6 和 2.7 相同):

def _some_str(value):
    try:
        return str(value)
    except Exception:
        pass
    try:
        value = unicode(value)
        return value.encode("ascii", "backslashreplace")
    except Exception:
        pass
    return '<unprintable %s object>' % type(value).__name__

如您所见,来自调用或调用的任何异常都会在该过程中停止,并且只给出“神秘”的表示。str()unicode.encode()

回溯模块与 Python 解释器的注意事项

traceback文档告诉我们的相反:

它在打印堆栈跟踪时完全模仿 Python 解释器的行为。

Python 解释器给出的表示在这里略有不同。与“不可打印”消息相反,解释器将简单地显示异常的名称,同时停止任何实际的异常。

这是一个简单的脚本,演示了所有三种方法:将异常留给 Python 解释器,使用 traceback 模块,或手动调用函数。

#!/usr/bin/python

import sys, traceback

class Boom(Exception):

    def __init__(self, foo, bar, baz):
        self.foo, self.bar, self.baz = foo, bar, baz

    def __str__(self):
        return ("boom! foo: %s, bar: %s, baz: "     # ouch! forgot an %s!
                % (self.foo, self.bar, self.baz))

def goBoom(): raise Boom(foo='FOO', bar='BAR', baz='BAZ')

if __name__ == "__main__":

    if sys.argv[1].startswith("i"):
        goBoom()
        # __main__.Boom

    elif sys.argv[1].startswith("t"):
        try:    goBoom()
        except: traceback.print_exc(file=sys.stdout)
        # Boom: <unprintable Boom object>

    elif sys.argv[1].startswith("m"):
        e = Boom(foo='FOO', bar='BAR', baz='BAZ')
        e.__str__()
        # TypeError: not all arguments converted during string formatting

    else: pass
于 2013-09-23T13:43:45.047 回答
8

我的猜测是你有 unicodefile_pathlist_lines变量,因为它没有被打印在没有 unicode 功能的控制台上。

或任何其他异常都__str__可能导致这种奇怪的行为,最好的方法是捕获异常并查看发生了什么,也使用调试器

def __str__(self):
    try:
        s =  'File {0} contains nested comments at lines {1}'.format(self.file_path, ', '.join(self.list_lines))
    except Exception,e:
        print "-----",type(e),e
    return s
于 2012-10-26T15:32:25.437 回答