百度360必应搜狗淘宝本站头条
当前位置:网站首页 > IT知识 > 正文

一文带您了解PySnooper:让码农告别Python print函数

liuian 2025-05-28 18:44 81 浏览

在我们之前的文章《一文带您了解IceCream:码农的Python代码调试神器,告别print()》,我们介绍了IceCream替代print进行代码调试的示例。今天,受到粉丝推荐的启发,我们将介绍另一个类似的Python代码调试包:PySnooper。

在编程过程中,调试通常是一项痛苦的任务,而使用传统的print语句进行调试往往更加繁琐。为了摆脱这种痛点,我们将直接介绍PySnooper,一款强大的代码调试工具。

我们首先看看PySnooper 官方介绍:https://github.com/cool-RR/PySnooper

PySnooper:再也不用为调试而使用print了

PySnooper是一个简易的调试工具。如果你曾经使用过Bash,它就像是Python的set -x命令,只不过更加高级。

你的情景:你试图弄清楚为什么你的Python代码不按照你预期的那样工作。你很想使用一个带有断点和监视功能的完整调试器,但是你现在懒得设置一个。

你想知道哪些代码行在运行,哪些没有运行,以及本地变量的值是什么。

大多数人会在关键位置使用print语句,其中一些显示变量的值。

PySnooper让你可以做同样的事情,但是你不需要精心编写正确的print语句,只需要在你感兴趣的函数上添加一个装饰器。你将得到你的函数的逐步日志,包括哪些行运行了以及何时运行,以及本地变量何时发生了变化。

PySnooper的独特之处在于什么?你可以在你那些糟糕、庞大的企业代码库中使用它,而无需进行任何设置。只需将装饰器添加到函数上,并将输出重定向到一个专用的日志文件,指定其路径作为第一个参数即可。

安装PySnooper

pip install pysnooper

第一个示例

我们正在编写一个将数字转换为二进制的函数,通过返回一个位列表来完成。让我们通过添加@pysnooper.snoop()装饰器来进行监视:

import pysnooper


@pysnooper.snoop()
def number_to_bits(number):
    if number:
        bits = []
        while number:
            number, remainder = divmod(number, 2)
            bits.insert(0, remainder)
        return bits
    else:
        return [0]


number_to_bits(6)

如果你不想追踪整个函数,你可以将相关部分包装在一个with块中:

import pysnooper
import random


def foo():
    lst = []
    for i in range(10):
        lst.append(random.randrange(1, 1000))


    with pysnooper.snoop():
        lower = min(lst)
        upper = max(lst)
        mid = (lower + upper) / 2
        print(lower, mid, upper)


foo()

PySnooper特性

PySnooper拥有许多功能,使其成为调试Python代码的一款多才多艺且实用的工具。以下是一些您应该了解和使用的顶级特性:

  • 定制PySnooper的输出。您可以通过在装饰器中使用不同的参数来定制PySnooper的输出,例如depth、prefix、file、watch等。例如,您可以使用depth来限制PySnooper跟踪的嵌套调用数量,或者使用watch来添加您想要监视的表达式。您可以在文档中找到参数的完整列表和描述。
  • 在任何Python代码、环境或框架中使用PySnooper。您可以在任何Python代码中使用PySnooper,包括第三方库、模块和包。您还可以在任何Python环境中使用它,例如Jupyter笔记本、脚本或交互式shell。您还可以在任何Python框架中使用PySnooper,例如Django、Flask、PyTorch、TensorFlow、PyTest、PyInstaller、PyGame、NumPy、Pandas、Scrapy、Matplotlib、SQLAlchemy等等。
  • 在多线程、多进程或异步代码中使用PySnooper。您可以通过在装饰器中使用thread_info或process_info参数来使用PySnooper调试多线程或多进程代码。这将在输出中添加线程或进程ID,以便您区分不同的线程或进程。您还可以使用PySnooper调试异步代码,只需使用@pysnooper.snoop_async()装饰器,而不是常规的装饰器。这将与任何异步框架一起使用,例如asyncio、trio或curio。
  • 在lambda函数、生成器、推导式和其他复杂表达式中使用PySnooper。您可以通过使用pysnooper.snoop函数而不是装饰器来调试lambda函数、生成器、推导式和其他复杂表达式。这将返回一个包装函数,您可以使用与原始函数相同的参数调用它。

像这样调试一个lambda函数:

import pysnooper


squared = pysnooper.snoop()(lambda x: x ** 2)
squared(5)

将日志输出到文件:

import pysnooper


@pysnooper.snoop('file.log')
def number_to_bits(number):
    if number:
        bits = []
        while number:
            number, remainder = divmod(number, 2)
            bits.insert(0, remainder)
        return bits
    else:
        return [0]


number_to_bits(6)

设置环境变量PYSNOOPER_DISABLED 为1 ,关闭debug 输出:

import os
os.environ['PYSNOOPER_DISABLED'] = '1'


import pysnooper


@pysnooper.snoop()
def factorial(n):
    if n == 1:
        return 1
    else:
        return n * factorial(n - 1)


factorial(5)

以一个前缀开头,方便使用grep查找所有的snoop行:

import pysnooper


@pysnooper.snoop(prefix='SNB ')
def factorial(n):
    if n == 1:
        return 1
    else:
        return n * factorial(n - 1)


factorial(5)

删除所有与机器相关的数据(路径、时间戳、内存地址),以便轻松地与其他跟踪进行比较:

import pysnooper


@pysnooper.snoop(normalize=True)
def factorial(n):
    if n == 1:
        return 1
    else:
        return n * factorial(n - 1)


factorial(5)

PySnooper的优势

PySnooper拥有许多优势,使其成为调试Python代码的绝佳工具。以下是其中一些:

  • PySnooper易于使用,无需设置或配置。您只需要导入它,并在您想要调试的函数或方法上添加一个装饰器。您无需学习任何命令或语法,也无需以任何方式修改您的代码。
  • PySnooper速度快且轻量级。它不会减慢您的代码,也不会增加任何额外的开销。它只会打印出您调试代码所需的相关信息,而不会多余打印。
  • PySnooper灵活且可定制。您可以通过在装饰器中使用不同的参数来调整PySnooper的输出,例如depth、prefix、file、watch等。您还可以在任何Python代码、环境或框架中使用PySnooper,包括多线程、多进程和异步代码。
  • PySnooper具有教育性和趣味性。它可以帮助您了解您的代码如何工作,找到错误,优化性能,并从其他人的代码中学习。它还可以使调试更加愉快,减少沮丧感,因为它向您展示了代码背后的魔力。


PySnooper是一款强大的Python代码调试工具,易于使用且无需配置。它能快速精准地定位问题,定制输出内容,并适用于各种Python环境和框架。通过添加装饰器,您可以轻松监视函数执行情况,而无需繁琐的print语句。PySnooper的灵活性和教育性使得调试变得更加高效愉快,为开发者提供了更好的调试体验。

相关推荐

教你把多个视频合并成一个视频的方法

一.情况介绍当你有一个m3u8文件和一个目录,目录中有连续的视频片段,这些片段可以连成一段完整的视频。m3u8文件打开后像这样:m3u8文件,可以理解为播放列表,里面是播放视频片段的顺序。视频片段像这...

零代码编程:用kimichat合并一个文件夹下的多个文件

一个文件夹里面有很多个srt字幕文件,如何借助kimichat来自动批量合并呢?在kimichat对话框中输入提示词:你是一个Python编程专家,完成如下的编程任务:这个文件夹:D:\downloa...

Java APT_java APT 生成代码

JavaAPT(AnnotationProcessingTool)是一种在Java编译阶段处理注解的工具。APT会在编译阶段扫描源代码中的注解,并根据这些注解生成代码、资源文件或其他输出,...

Unit Runtime:一键运行 AI 生成的代码,或许将成为你的复制 + 粘贴神器

在我们构建了UnitMesh架构之后,以及对应的demo之后,便着手于实现UnitMesh架构。于是,我们就继续开始UnitRuntime,以用于直接运行AI生成的代码。PS:...

挣脱臃肿的枷锁:为什么说Vert.x是Java开发者手中的一柄利剑?

如果你是一名Java开发者,那么你的职业生涯几乎无法避开Spring。它如同一位德高望重的老国王,统治着企业级应用开发的大片疆土。SpringBoot的约定大于配置、SpringCloud的微服务...

五年后,谷歌还在全力以赴发展 Kotlin

作者|FredericLardinois译者|Sambodhi策划|Tina自2017年谷歌I/O全球开发者大会上,谷歌首次宣布将Kotlin(JetBrains开发的Ja...

kotlin和java开发哪个好,优缺点对比

Kotlin和Java都是常见的编程语言,它们有各自的优缺点。Kotlin的优点:简洁:Kotlin程序相对于Java程序更简洁,可以减少代码量。安全:Kotlin在类型系统和空值安全...

移动端架构模式全景解析:从MVC到MVVM,如何选择最佳设计方案?

掌握不同架构模式的精髓,是构建可维护、可测试且高效移动应用的关键。在移动应用开发中,选择合适的软件架构模式对项目的可维护性、可测试性和团队协作效率至关重要。随着应用复杂度的增加,一个良好的架构能够帮助...

颜值非常高的XShell替代工具Termora,不一样的使用体验!

Termora是一款面向开发者和运维人员的跨平台SSH终端与文件管理工具,支持Windows、macOS及Linux系统,通过一体化界面简化远程服务器管理流程。其核心定位是解决多平台环境下远程连接、文...

预处理的底层原理和预处理编译运行异常的解决方案

若文章对您有帮助,欢迎关注程序员小迷。助您在编程路上越走越好![Mac-10.7.1LionIntel-based]Q:预处理到底干了什么事情?A:预处理,顾名思义,预先做的处理。源代码中...

为“架构”再建个模:如何用代码描述软件架构?

在架构治理平台ArchGuard中,为了实现对架构的治理,我们需要代码+模型描述所要处理的内容和数据。所以,在ArchGuard中,我们有了代码的模型、依赖的模型、变更的模型等,剩下的两个...

深度解析:Google Gemma 3n —— 移动优先的轻量多模态大模型

2025年6月,Google正式发布了Gemma3n,这是一款能够在2GB内存环境下运行的轻量级多模态大模型。它延续了Gemma家族的开源基因,同时在架构设计上大幅优化,目标是让...

比分网开发技术栈与功能详解_比分网有哪些

一、核心功能模块一个基本的比分网通常包含以下模块:首页/总览实时比分看板:滚动展示所有正在进行的比赛,包含比分、比赛时间、红黄牌等关键信息。热门赛事/焦点战:突出显示重要的、关注度高的比赛。赛事导航...

设计模式之-生成器_一键生成设计

一、【概念定义】——“分步构建复杂对象,隐藏创建细节”生成器模式(BuilderPattern):一种“分步构建型”创建型设计模式,它将一个复杂对象的构建与其表示分离,使得同样的构建过程可以创建...

构建第一个 Kotlin Android 应用_kotlin简介

第一步:安装AndroidStudio(推荐IDE)AndroidStudio是官方推荐的Android开发集成开发环境(IDE),内置对Kotlin的完整支持。1.下载And...