编写MindSpore数据集加载代码的时候,有时候会遇到一些迷惑的代码行为,比如不知道代码卡在哪里不动了,似乎处理的很慢,或者写了死循环的逻辑。这些问题往往会花费很多精力打点去调试python代码,那么有没有方便的工具帮我们这么做呢
—— 有请今天的主角:py-spy
py-spy 介绍
引用官方的话:
py-spy是Python程序的采样分析器。它可以让您可视化Python程序花费的时间在什么上,而无需重新启动程序或以任何方式修改代码。py-spy开销极低:它是用Rust编写的,以提高速度,并且与分析的Python程序不在相同的进程中运行。这意味着py-spy可以安全地用于生产Python代码。
官方github地址:https://github.com/benfred/py-spy 如果有相关的问题也可以去github提问。
py-spy 安装
pip install py-spy
安装后使用 py-spy -h
可以验证安装,并查看使用帮助。
py-spy -h
py-spy 0.3.12
Sampling profiler for Python programs
USAGE:
py-spy <SUBCOMMAND>
OPTIONS:
-h, --help Print help information
-V, --version Print version information
SUBCOMMANDS:
record Records stack trace information to a flamegraph, speedscope or raw file
top Displays a top like view of functions consuming CPU
dump Dumps stack traces for a target program to stdout
help Print this message or the help of the given subcommand(s)
py-spy 基本功能
一般在命令行中运行py-spy,并对py-spy传入要分析的进程的PID或要运行的python程序。
py-spy具有三个子命令record,top和dump:
- record生成火焰图
- top实时查看每个函数运行时间并统计
- dump显示每个python线程的当前调用堆栈
使用py-spy调试MindSpore数据集加载
构造一个数据迭代很慢的场景
先上一个经典代码
test_dataset.py
import mindspore.dataset as ds
import numpy as np
import time
class DatasetGenerator:
def __init__(self):
pass
def __getitem__(self, item):
self.do_something()
return (np.array([item]),)
def do_something(self):
cnt = 0
for i in range(100000000):
cnt += 1
def __len__(self):
return 50
def test_generator_0():
data1 = ds.GeneratorDataset(DatasetGenerator(), ["data"])
start = time.time()
for item in data1.create_dict_iterator(num_epochs=1, output_numpy=True):
print("data time:", time.time() - start)
start = time.time()
虽然这个用例比较简单,但是确实会看到迭代的时候出数据很慢,运行一下可以看到
data time: 5.431891679763794
data time: 5.6114866733551025
data time: 5.38549542427063
data time: 5.577831268310547
基本上5秒才读出一条数据,50条数据就是250秒!这才50条数据,如果10w条呢,这个读取时间难以接受。
那么究竟问题出现在哪里呢?(虽然明眼能看到do_something这个函数有点问题,但是我们现在假装不知道)
我们需要py-spy帮我们定位一下问题。
使用py-spy的top功能查看调用栈和时间占比
我们重新运行一下这个脚本
python test_dataset.py &
注意我们允许的时候加了 &
使其可以在后台运行,同时返回一个pid,比如这样
我们得到了这个python的进程id 116079,有了这个pid之后,我们就可以利用py-spy给我们做实时分析了,接下来我们在命令行中输入
py-spy top --pid 116079
假如遇到这个提示
Permission Denied: Try running again with elevated permissions by going 'sudo env "PATH=$PATH" !!'
重新在命令行输入这个即可
sudo env "PATH=$PATH" !!
此时将会看到py-spy列出了调用栈及相关的函数用时占比:
直观地从这个调用栈分析,do_something这个函数居然占用了100%的运行时间,运行超过了50+s,肯定有某些问题!或是处理逻辑太慢,或是写了死循环?
那么我们就可以快速来到代码的这里看看发生什么事了
def do_something(self):
cnt = 0
for i in range(100000000):
cnt += 1
天哪!这里居然有一个一亿次的循环加法,究竟是谁写的代码!(不是我) 我们可以通过优化这一部分代码,来提升整体数据处理的速度。
总的来说
py-spy是一个挺方便的工具,帮助用户快速定位python的性能问题,除了top功能之外,py-spy也提供了dump功能把完整的堆栈写入到本地方便分析。当我们在 debug 一个线上正在运行的程序的时候,只需要提供进程id,py-spy就可以直接接入使用。