"pdb" --- Python 的调试器
*************************

**源代码：** Lib/pdb.py

======================================================================

"pdb" 模块定义了一个交互式源代码调试器，用于 Python 程序。它支持在源码
行间设置（有条件的）断点和单步执行，检视堆栈帧，列出源码列表，以及在任
何堆栈帧的上下文中运行任意 Python 代码。它还支持事后调试，可以在程序控
制下调用。

调试器是可扩展的——调试器实际被定义为 "Pdb" 类。该类目前没有文档，但通
过阅读源码很容易理解它。扩展接口使用了 "bdb" 和 "cmd" 模块。

参见:

  模块 "faulthandler"
     用于在发生错误、超时或用户信号时显式地转储 Python 回溯信息。

  模块 "traceback"
     提取、格式化和打印 Python 程序的栈回溯信息的标准接口。

中断进入调试器的典型用法是插入:

   import pdb; pdb.set_trace()

或者:

   breakpoint()

到你想进入调试器的位置，再运行程序。 然后你可以单步执行这条语句之后的
代码，并使用 "continue" 命令来关闭调试器继续运行。

在 3.7 版本发生变更: 内置函数 "breakpoint()"，当以默认方式调用时，可被
用来代替 "import pdb; pdb.set_trace()"。

   def double(x):
      breakpoint()
      return x * 2
   val = 3
   print(f"{val} * 2 is {double(val)}")

调试器的提示符为 "(Pdb)"，这指明你正处于调试模式下:

   > ...(2)double()
   -> breakpoint()
   (Pdb) p x
   3
   (Pdb) continue
   3 * 2 is 6

在 3.3 版本发生变更: 由 "readline" 模块实现的 Tab 补全可用于补全本模块
的命令和命令的参数，例如，Tab 补全会提供当前的全局变量和局部变量，用作
"p" 命令的参数。

你还可以从命令行发起调用 "pdb" 来调试其他脚本。 例如:

   python -m pdb myscript.py

当作为模块发起调用时，如果被调试的程序异常退出则 pdb 将自动进入事后调
试。 在事后调试之后（或程序正常退出之后），pdb 将重启程序。 自动重启会
保留 pdb 的状态（如断点）并且在大多数情况下这比在退出程序的同时退出调
试器更实用。

在 3.2 版本发生变更: 增加了 "-c" 选项用来执行如同在 ".pdbrc" 文件中给
出的命令；参见 调试器命令。

在 3.7 版本发生变更: 增加了 "-m" 选项用来以类似 "python -m" 的方式来执
行模块。 就像一个脚本那样，调试器将在模块的第一行之前暂停执行。

在调试器控制下执行一条语句的典型用法如下:

   >>> import pdb
   >>> def f(x):
   ...     print(1 / x)
   >>> pdb.run("f(2)")
   > <string>(1)<module>()
   (Pdb) continue
   0.5
   >>>

检查已崩溃程序的典型用法是:

   >>> import pdb
   >>> def f(x):
   ...     print(1 / x)
   ...
   >>> f(0)
   Traceback (most recent call last):
     File "<stdin>", line 1, in <module>
     File "<stdin>", line 2, in f
   ZeroDivisionError: division by zero
   >>> pdb.pm()
   > <stdin>(2)f()
   (Pdb) p x
   0
   (Pdb)

在 3.13 版本发生变更: **PEP 667** 的实现意味着通过 "pdb" 进行的变量名
赋值将立刻影响当前作用域，即使是在 *optimized scope* 中运行的时候。

本模块定义了下列函数，每个函数进入调试器的方式略有不同：

pdb.run(statement, globals=None, locals=None)

   在调试器控制范围内执行 *statement* （以字符串或代码对象的形式提供）
   。调试器提示符会在执行代码前出现，你可以设置断点并键入 "continue"，
   也可以使用 "step" 或 "next" 逐步执行语句（上述所有命令在后文有说明
   ）。可选参数 *globals* 和 *locals* 指定代码执行环境，默认时使用
   "__main__" 模块的字典。（请参阅内置函数 "exec()" 或 "eval()" 的说明
   。）

pdb.runeval(expression, globals=None, locals=None)

   在调试器控制下对 *expression* (以字符串或代码对象的形式给出) 求值。
   当 "runeval()" 返回时，它将返回 *expression* 的值。 在其他方面此函
   数与 "run()" 类似。

pdb.runcall(function, *args, **kwds)

   使用给定的参数调用 *function* （以函数或方法对象的形式提供，不能是
   字符串）。"runcall()" 返回的是所调用函数的返回值。调试器提示符将在
   进入函数后立即出现。

pdb.set_trace(*, header=None)

   在调用本函数的堆栈帧处进入调试器。用于硬编码一个断点到程序中的固定
   点处，即使该代码不在调试状态（如断言失败时）。如果传入 *header*，它
   将在调试开始前被打印到控制台。

   在 3.7 版本发生变更: 仅关键字参数 *header*。

   在 3.13 版本发生变更: "set_trace()" 将立即进入调试器，而不是在下一
   行要执行的代码上进入。

pdb.post_mortem(traceback=None)

   进入 *traceback* 对象的事后调试。如果没有给定 *traceback*，默认使用
   当前正在处理的异常之一（默认时，必须存在正在处理的异常）。

pdb.pm()

   进入在 "sys.last_exc" 中找到的异常的事后调试。

"run*" 函数和 "set_trace()" 都是别名，用于实例化 "Pdb" 类和调用同名方
法。如果要使用其他功能，则必须自己执行以下操作：

class pdb.Pdb(completekey='tab', stdin=None, stdout=None, skip=None, nosigint=False, readrc=True)

   "Pdb" 是调试器类。

   *completekey*、*stdin* 和 *stdout* 参数都会传递给底层的 "cmd.Cmd"
   类，请参考相应的描述。

   如果给出 *skip* 参数，则它必须是一个迭代器，可以迭代出 glob-style
   样式的模块名称。如果遇到匹配上述样式的模块，调试器将不会进入来自该
   模块的堆栈帧。 [1]

   默认情况下，当发出 "continue" 命令时，Pdb 将为 SIGINT 信号（信号当
   用户在控制台按 "Ctrl-C" 时发出的）设置一个处理器。 这使用户可以通过
   按 "Ctrl-C" 再次进入调试器。 如果你希望 Pdb 不要改变 SIGINT 处理器
   ，请将 *nosigint* 设为真值。to true.

   *readrc* 参数默认为 true，它控制 Pdb 是否从文件系统加载 .pdbrc 文件
   。

   启用跟踪且带有 *skip* 参数的调用示范:

      import pdb; pdb.Pdb(skip=['django.*']).set_trace()

   引发一个不带参数的 审计事件 "pdb.Pdb"。

   在 3.1 版本发生变更: 增加了 *skip* 形参。

   在 3.2 版本发生变更: 增加了 *nosigint* 形参。 在之前版本中，pdb 绝
   不会设置 SIGINT 处理器。

   在 3.6 版本发生变更: *readrc* 参数。

   run(statement, globals=None, locals=None)
   runeval(expression, globals=None, locals=None)
   runcall(function, *args, **kwds)
   set_trace()

      请参阅上文解释同名函数的文档。


调试器命令
==========

下方列出的是调试器可接受的命令。如下所示，大多数命令可以缩写为一个或两
个字母。如 "h(elp)" 表示可以输入 "h" 或 "help" 来输入帮助命令（但不能
输入 "he" 或 "hel"，也不能是 "H" 或 "Help" 或 "HELP"）。命令的参数必须
用空格（空格符或制表符）分隔。在命令语法中，可选参数括在方括号 ("[]")
中，使用时请勿输入方括号。命令语法中的选择项由竖线 ("|") 分隔。

输入一个空白行将重复最后输入的命令。例外：如果最后一个命令是 "list" 命
令，则会列出接下来的 11 行。

调试器无法识别的命令将被认为是 Python 语句，并在正在调试的程序的上下文
中执行。Python 语句也可以用感叹号 ("!") 作为前缀。这是检查正在调试的程
序的强大方法，甚至可以修改变量或调用函数。当此类语句发生异常，将打印异
常名称，但调试器的状态不会改变。

在 3.13 版本发生变更: 前缀为 pdb 命令的表达式/语句现在会被正确地标识并
执行。

调试器支持 别名。别名可以有参数，使得调试器对被检查的上下文有一定程度
的适应性。

在一行中可以输入多条命令，以 ";;" 分隔。 （不能使用单个 ";"，因为它已
被用作传给 Python 解析器的一行中的多条命令的分隔符。） 命令切分所用的
方式没有任何智能可言；输入总是会在第一个 ";;" 对上被切分，即使它位于带
引号的字符串中。 对于带有双分号的字符串可以使用隐式字符串拼接 "';'';'"
或 "";"";"" 来变通处理。

要设置临时全局变量，请使用 *快捷变量*。 *快捷变量* 是名称以 "$" 打头的
变量。 例如，"$foo = 1" 将设置一个全局变量 "$foo" 供你在调试器会话中使
用。 *快捷变量* 会在程序恢复执行时被清空因此它不大可能像使用普通变量如
"foo = 1" 那样影响到你的程序。

存在三个预设的 *快捷变量*:

* "$_frame": 你正在调试的当前帧

* "$_retval": 当帧返回时的返回值

* "$_exception": 当帧引发异常时的异常值

Added in version 3.12: 增加了 *便捷变量* 特性。

如果文件 ".pdbrc" 存在于用户主目录或当前目录中，则它将以 "'utf-8'" 编
码格式被读入并执行，就像是在调试器提示符下被键入一样，不同之处在于空行
和以 "#" 开头的行会被忽略。 这对于别名特别有用。 如果两个文件都存在，
则会先读取主目录中的文件并且在那里定义的别名可以被本地文件所覆盖。

在 3.2 版本发生变更: ".pdbrc" 现在可以包含继续调试的命令，如
"continue" 或 "next"。文件中的这些命令以前是无效的。

在 3.11 版本发生变更: ".pdbrc" 现在将以 "'utf-8'" 编码格式来读取。 在
之前版本中，它是以系统语言区域编码格式来读取的。

h(elp) [command]

   不带参数时，显示可用的命令列表。参数为 *command* 时，打印有关该命令
   的帮助。"help pdb" 显示完整文档（即 "pdb" 模块的文档字符串）。由于
   *command* 参数必须是标识符，因此要获取 "!" 的帮助必须输入 "help
   exec"。

w(here)

   打印栈回溯，最新的帧位于底部。 有一个箭头 (">") 指明当前帧，该帧决
   定了大多数命令的上下文。

d(own) [count]

   在堆栈回溯中，将当前帧向下移动 *count* 级（默认为 1 级，移向更新的
   帧）。

u(p) [count]

   在堆栈回溯中，将当前帧向上移动 *count* 级（默认为 1 级，移向更老的
   帧）。

b(reak) [([filename:]lineno | function) [, condition]]

   传入 *lineno* 参数，在当前文件内的第 *lineno* 行设置中断。 行号数值
   开头可以带有 *filename* 加一个冒号，以在另一个文件内指定中断点（可
   能是尚未载入的文件）。 文件将根据 "sys.path" 来搜索。 可接受的
   *filename* 形式有 "/abspath/to/file.py", "relpath/file.py",
   "module" 和 "package.module"。

   传入 *function* 参数，在该函数内的第一条可执行语句上设置中断。
   *function* 可以是会在当前命名空间中被求值为一个函数的任意表达式。

   如果第二个参数存在，它应该是一个表达式，且它的计算值为 true 时断点
   才起作用。

   如果不带参数执行，将列出所有中断，包括每个断点、命中该断点的次数、
   当前的忽略次数以及关联的条件（如果有）。

   每个中断点将被分配一个数值供所有其他中断点命令引用。

tbreak [([filename:]lineno | function) [, condition]]

   临时断点，在第一次命中时会自动删除。它的参数与 "break" 相同。

cl(ear) [filename:lineno | bpnumber ...]

   如果参数是 *filename:lineno*，则清除此行上的所有断点。如果参数是空
   格分隔的断点编号列表，则清除这些断点。如果不带参数，则清除所有断点
   （但会先提示确认）。

disable bpnumber [bpnumber ...]

   禁用断点，断点以空格分隔的断点编号列表给出。禁用断点表示它不会导致
   程序停止执行，但是与清除断点不同，禁用的断点将保留在断点列表中并且
   可以（重新）启用。

enable bpnumber [bpnumber ...]

   启用指定的断点。

ignore bpnumber [count]

   为指定的断点编号设置忽略次数。 如果省略 *count*，则忽略次数将设置为
   0。 当忽略次数为零时断点将变为活动状态。 如果为非零值，则在每次到达
   断点且断点未禁用且关联条件取真值时 *count* 就像递减。

condition bpnumber [condition]

   为断点设置一个新 *condition*，它是一个表达式，且它的计算值为 true
   时断点才起作用。如果没有给出 *condition*，则删除现有条件，也就是将
   断点设为无条件。

commands [bpnumber]

   为编号是 *bpnumber* 的断点指定一系列命令。命令内容将显示在后续的几
   行中。输入仅包含 "end" 的行来结束命令列表。举个例子:

      (Pdb) commands 1
      (com) p some_variable
      (com) end
      (Pdb)

   要删除断点上的所有命令，请输入 "commands" 并立即以 "end" 结尾，也就
   是不指定任何命令。

   如果不带 *bpnumber* 参数，"commands" 作用于最后一个被设置的断点。

   可以为断点指定命令来重新启动程序。只需使用 "continue" 或 "step" 命
   令或其他可以继续运行程序的命令。

   如果指定了某个继续运行程序的命令（目前包括 "continue", "step",
   "next", "return", "jump", "quit" 及它们的缩写）将终止命令列表（就像
   该命令后紧跟着 end）。因为在任何时候继续运行下去（即使是简单的 next
   或 step），都可能会遇到另一个断点，该断点可能具有自己的命令列表，这
   导致要执行的列表含糊不清。

   如果在命令列表中使用 "silent" 命令，那么在断点处停下时就不会打印常
   规信息。 这正是要打印特定消息然后继续运行的断点所想要的。 如果没有
   其他命令来打印任何消息，则你将不会看到已到达断点的迹象。

s(tep)

   运行当前行，在第一个可以停止的位置（在被调用的函数内部或在当前函数
   的下一行）停下。

n(ext)

   继续运行，直到运行到当前函数的下一行，或当前函数返回为止。（ "next"
   和 "step" 之间的区别在于，"step" 进入被调用函数内部并停止，而
   "next" （几乎）全速运行被调用函数，仅在当前函数的下一行停止。）

unt(il) [lineno]

   如果不带参数，则继续运行，直到行号比当前行大时停止。

   如果带有 *lineno*，则继续执行直至行号大于或等于 *lineno*。 在这两种
   情况下，在当前帧返回时也将停止。

   在 3.2 版本发生变更: 允许明确给定行号。

r(eturn)

   继续运行，直到当前函数返回。

c(ont(inue))

   继续运行，仅在遇到断点时停止。

j(ump) lineno

   设置即将运行的下一行。仅可用于堆栈最底部的帧。它可以往回跳来再次运
   行代码，也可以往前跳来跳过不想运行的代码。

   需要注意的是，不是所有的跳转都是允许的 -- 例如，不能跳转到 "for" 循
   环的中间或跳出 "finally" 子句。

l(ist) [first[, last]]

   列出当前文件的源代码。如果不带参数，则列出当前行周围的 11 行，或延
   续前一次列出。如果用 "." 作为参数，则列出当前行周围的 11 行。如果带
   有一个参数，则列出那一行周围的 11 行。如果带有两个参数，则列出所给
   的范围中的代码；如果第二个参数小于第一个参数，则将其解释为列出行数
   的计数。

   当前帧中的当前行用 "->" 标记。如果正在调试异常，且最早抛出或传递该
   异常的行不是当前行，则那一行用 ">>" 标记。

   在 3.2 版本发生变更: 增加了 ">>" 标记。

ll | longlist

   列出当前函数或帧的所有源代码。相关行的标记与 "list" 相同。

   Added in version 3.2.

a(rgs)

   打印当前函数的参数及其当前的值。

p expression

   在当前上下文中对 *expression* 求值并打印该值。

   备注:

     "print()" 也可以使用，但它不是一个调试器命令 --- 它执行 Python
     "print()" 函数。

pp expression

   与 "p" 命令类似，但 *expression* 的值将使用 "pprint" 模块美观地打印
   。

whatis expression

   打印 *expression* 的类型。

source expression

   尝试获取 *expression* 的源代码并显示它。

   Added in version 3.2.

display [expression]

   如果 *expression* 的值发生变化则显示它的值，每次都会停止执行当前帧
   。

   如果不带 *expression*，则列出当前帧的所有显示表达式。

   备注:

     显示 *expression* 的值并与 *expression* 之前的求值结果进行比较，
     因此当结果可变时，显示可能无法体现变化。

   示例：

      lst = []
      breakpoint()
      pass
      lst.append(1)
      print(lst)

   显示将不会发现 "lst" 已被改变因为求值结果在执行比较之前已被
   "lst.append(1)" 原地修改了:

      > example.py(3)<module>()
      -> pass
      (Pdb) display lst
      display lst: []
      (Pdb) n
      > example.py(4)<module>()
      -> lst.append(1)
      (Pdb) n
      > example.py(5)<module>()
      -> print(lst)
      (Pdb)

   你可以通过拷贝机制巧妙地实现此功能:

      > example.py(3)<module>()
      -> pass
      (Pdb) display lst[:]
      display lst[:]: []
      (Pdb) n
      > example.py(4)<module>()
      -> lst.append(1)
      (Pdb) n
      > example.py(5)<module>()
      -> print(lst)
      display lst[:]: [1]  [old: []]
      (Pdb)

   Added in version 3.2.

undisplay [expression]

   不再显示当前帧中的 *expression*。 如果不带 *expression*，则清除当前
   帧的所有显示表达式。

   Added in version 3.2.

interact

   在根据当前作用域的局部和全局命名空间初始化的新建全局命名空间中启动
   一个交互式解释器 (使用 "code" 模块)。 使用 "exit()" 或 "quit()" 退
   出解释器并返回调试器。

   备注:

     由于 "interact" 为代码执行创建了一个新的专用命名空间，对变量的赋
     值将不会影响原始命名空间。 不过，对任何被引用的可变对象的修改都将
     照常在原始命名空间中反映出来。

   Added in version 3.2.

   在 3.13 版本发生变更: 可以使用 "exit()" 和 "quit()" 退出 "interact"
   命令。

   在 3.13 版本发生变更: "interact" 会将其输出引向调试器的输出通道而不
   是 "sys.stderr"。

alias [name [command]]

   创建一个名为 *name* 的别名用来执行 *command*。 *command* 的两边 *不
   可* 用引号括起来。 可替换形参可以通过 "%1", "%2" ... 和 "%9" 等来指
   明，而 "%*" 将被所有这些形参替换。 如果省略 *command*，则将显示
   *name* 的当前别名。 如未给出任何参数，则将列出所有别名。

   别名允许嵌套并可包含能在 pdb 提示符下合法输入的任何内容。 请注意内
   部 pdb 命令 *可以* 被别名所覆盖。 这样的命令将被隐藏直到别名被移除
   。 别名会递归地应用到命令行的第一个单词；行内的其他单词不会受影响。

   作为示例，这里列出了两个有用的别名（特别适合放在 ".pdbrc" 文件中）:

      # Print instance variables (usage "pi classInst")
      alias pi for k in %1.__dict__.keys(): print(f"%1.{k} = {%1.__dict__[k]}")
      # Print instance variables in self
      alias ps pi self

unalias name

   删除指定的别名 *name*。

! statement

   在当前栈帧的上下文中执行 (单行的) *statement*。 感叹号可以被省略，
   除非第一个语句的第一个单词与某个调试器命名重名，例如:

      (Pdb) ! n=42
      (Pdb)

   要设置全局变量，你可以在同一行上在赋值命令前添加 "global" 语句，例
   如:

      (Pdb) global list_options; list_options = ['-l']
      (Pdb)

run [args ...]
restart [args ...]

   重启被调试的 Python 程序。 如果提供了 *args*，它会用 "shlex" 来拆分
   且拆分结果将被用作新的 "sys.argv"。 历史、中断点、动作和调试器选项
   将被保留。 "restart" 是 "run" 的一个别名。

q(uit)

   退出调试器。 被执行的程序将被中止。

debug code

   进入一个对 *code* 执行步进的递归调试器（该参数是在当前环境中执行的
   任意表达式或语句）。

retval

   打印当前函数最后一次返回的返回值。

exceptions [excnumber]

   列出串连的异常或在其间跳转。

   当将 "pdb.pm()"  或 "Pdb.post_mortem(...)" 用于串连的异常而不是回溯
   数据时，它允许用户在串连的异常间移动，使用 "exceptions" 命令列出异
   常，并使用 "exception <number>" 切换到该异常。

   示例：

      def out():
          try:
              middle()
          except Exception as e:
              raise ValueError("reraise middle() error") from e

      def middle():
          try:
              return inner(0)
          except Exception as e:
              raise ValueError("Middle fail")

      def inner(x):
          1 / x

       out()

   调用 "pdb.pm()" 将允许在异常之间移动:

      > example.py(5)out()
      -> raise ValueError("reraise middle() error") from e

      (Pdb) exceptions
        0 ZeroDivisionError('division by zero')
        1 ValueError('Middle fail')
      > 2 ValueError('reraise middle() error')

      (Pdb) exceptions 0
      > example.py(16)inner()
      -> 1 / x

      (Pdb) up
      > example.py(10)middle()
      -> return inner(0)

   Added in version 3.13.

-[ 备注 ]-

[1] 一个帧是否会被认为源自特定模块是由帧全局变量 "__name__" 来决定的。
