网易首页 > 网易号 > 正文 申请入驻

如何快速定位 Python 运行最慢的代码?

0
分享至

△点击上方“Python猫”关注 ,回复“1”领取电子书

天下武功,唯快不破。

编程也不例外,你的代码跑的快,你能快速找出代码慢的原因,你的码功就高。

今天分享一个超级实用的 Python 性能分析工具 pyinstrument ,可以快速找到代码运行最慢的部分,帮助提高代码的性能。支持 Python 3.7+ 且能够分析异步代码,仅需一条命令即可显示具体代码的耗时。经常写 Python 的小伙伴一定要用一下。

安装pip install pyinstrument
简单的使用

在程序的开始,启动 pyinstrument 的 Profiler,结束时关闭 Profiler 并打印分析结果如下:

from pyinstrument import Profiler

profiler = Profiler()
profiler.start()

# 这里是你要分析的代码

profiler.stop()

profiler.print()

比如这段代码 123.py,我们可以清楚的看到是列表推导式比较慢:

from pyinstrument import Profiler

profiler = Profiler()
profiler.start()

# 这里是你要分析的代码
a = [i for i in range(100000)]
b = (i for i in range(100000))

profiler.stop()
profiler.print()

上述分析需要修改源代码,如果你使用命令行工具,就不需要修改源代码,只需要执行pyinstrument xxxx.py即可:

比如有这样一段排序的程序 c_sort.py:

import sys
import time

import numpy as np

arr = np.random.randint(0, 10, 10)

def slow_key(el):
time.sleep(0.01)
return el

arr = list(arr)

for i in range(10):
arr.sort(key=slow_key)

print(arr)

这段代码里面故意放了一句 time.sleep(0.01) 来延迟性能,看看pyinstrument能否识别,命令行执行pyinstrument c_sort.py:

从结果来看,程序运行了 1.313 秒,而 sleep 就运行了 1.219 秒,很明显是瓶颈,现在我们把它删除,再看看结果:

删除之后,性能最慢的就是 numpy 模块的初始化代码__init__.py了,不过这些代码不是自己写的,而且并不是特别慢,就不需要去关心了。

分析 Flask 代码

Web 应用也可以使用这个来找出性能瓶颈,比如 flask,只需要在请求之前记录时间,在请求之后统计时间,只需要在 flask 的请求拦截器里面这样写:

from flask import Flask, g, make_response, request
app = Flask(__name__)

@app.before_request
def before_request():
if "profile" in request.args:
g.profiler = Profiler()
g.profiler.start()

@app.after_request
def after_request(response):
if not hasattr(g, "profiler"):
return response
g.profiler.stop()
output_html = g.profiler.output_html()
return make_response(output_html)

假如有这样一个 API:

@app.route("/dosomething")
def do_something():
import requests
requests.get("http://google.com")
return "Google says hello!"

为了测试这个 API 的瓶颈,我们可以在 url 上加一个参数 profile 就可以:http://127.0.0.1:5000/dosomething?profile,哪一行代码执行比较慢,结果清晰可见:

分析 Django 代码也非常简单,只需要在 Django 的配置文件的 MIDDLEWARE 中添加

"pyinstrument.middleware.ProfilerMiddleware",

然后就可以在 url 上加一个参数 profile 就可以:

如果你不希望所有人都能看到,只希望管理员可以看到,settings.py 可以添加这样的代码:

def custom_show_pyinstrument(request):
return request.user.is_superuser

PYINSTRUMENT_SHOW_CALLBACK = "%s.custom_show_pyinstrument" % __name__

如果不想通过 url 后面加参数的方式查看性能分析,可以在 settings.py 文件中添加:

PYINSTRUMENT_PROFILE_DIR = 'profiles'

这样,每次访问一次 Django 接口,就会将分析结果以 html 文件形式保存在 项目目录下的 profiles 文件夹中。

分析异步代码

简单的异步代码分析:

async_example_simple.py:

import asyncio

from pyinstrument import Profiler

async def main():
p = Profiler()
with p:
print("Hello ...")
await asyncio.sleep(1)
print("... World!")
p.print()

asyncio.run(main())

复杂一些的异步代码分析:

import asyncio
import time

import pyinstrument

def do_nothing():
pass

def busy_wait(duration):
end_time = time.time() + duration

while time.time() < end_time:
do_nothing()

async def say(what, when, profile=False):
if profile:
p = pyinstrument.Profiler()
p.start()

busy_wait(0.1)
sleep_start = time.time()
await asyncio.sleep(when)
print(f"slept for {time.time() - sleep_start:.3f} seconds")
busy_wait(0.1)

print(what)
if profile:
p.stop()
p.print(show_all=True)

loop = asyncio.get_event_loop()

loop.create_task(say("first hello", 2, profile=True))
loop.create_task(say("second hello", 1, profile=True))
loop.create_task(say("third hello", 3, profile=True))

loop.run_forever()
loop.close()
工作原理

Pyinstrument 每 1ms 中断一次程序,并在该点记录整个堆栈。它使用 C 扩展名和 PyEval_SetProfile 来做到这一点,但只每 1 毫秒读取一次读数。你可能觉得报告的样本数量有点少,但别担心,它不会降低准确性。默认间隔 1ms 是记录堆栈帧的下限,但如果在单个函数调用中花费了很长时间,则会在该调用结束时进行记录。如此有效地将这些样本“打包”并在最后记录。

Pyinstrument 是一个统计分析器,并不跟踪,它不会跟踪您的程序进行的每个函数调用。相反,它每 1 毫秒记录一次调用堆栈。与其他分析器相比,统计分析器的开销比跟踪分析器低得多。

比如说,我想弄清楚为什么 Django 中的 Web 请求很慢。如果我使用 cProfile,我可能会得到这个:

151940 function calls (147672 primitive calls) in 1.696 seconds

Ordered by: cumulative time

ncalls tottime percall cumtime percall filename:lineno(function)
1 0.000 0.000 1.696 1.696 profile:0(
at 0x1053d6a30, file "./manage.py", line 2>)
1 0.001 0.001 1.693 1.693 manage.py:2()
1 0.000 0.000 1.586 1.586 __init__.py:394(execute_from_command_line)
1 0.000 0.000 1.586 1.586 __init__.py:350(execute)
1 0.000 0.000 1.142 1.142 __init__.py:254(fetch_command)
43 0.013 0.000 1.124 0.026 __init__.py:1()
388 0.008 0.000 1.062 0.003 re.py:226(_compile)
158 0.005 0.000 1.048 0.007 sre_compile.py:496(compile)
1 0.001 0.001 1.042 1.042 __init__.py:78(get_commands)
153 0.001 0.000 1.036 0.007 re.py:188(compile)
106/102 0.001 0.000 1.030 0.010 __init__.py:52(__getattr__)
1 0.000 0.000 1.029 1.029 __init__.py:31(_setup)
1 0.000 0.000 1.021 1.021 __init__.py:57(_configure_logging)
2 0.002 0.001 1.011 0.505 log.py:1()

看完是不是还是一脸懵逼,通常很难理解您自己的代码如何与这些跟踪相关联。Pyinstrument 记录整个堆栈,因此跟踪昂贵的调用要容易得多。它还默认隐藏库框架,让您专注于影响性能的应用程序/模块:

_ ._ __/__ _ _ _ _ _/_ Recorded: 14:53:35 Samples: 131
/_//_/// /_\ / //_// / //_'/ // Duration: 3.131 CPU time: 0.195
/ _/ v3.0.0b3

Program: examples/django_example/manage.py runserver --nothreading --noreload

3.131 manage.py:2
└─ 3.118 execute_from_command_line django/core/management/__init__.py:378
[473 frames hidden] django, socketserver, selectors, wsgi...
2.836 select selectors.py:365
0.126 _get_response django/core/handlers/base.py:96
└─ 0.126 hello_world django_example/views.py:4
最后的话

本文分享了 pyinstrument 的用法,有了这个性能分析神器,以后优化代码可以节省很多时间了,这样的效率神器很值得分享,毕竟人生苦短,能多点时间干点有意思的不香么?

还不过瘾?试试它们

如果你觉得本文有帮助

请慷慨分享点赞,感谢啦

特别声明:以上内容(如有图片或视频亦包括在内)为自媒体平台“网易号”用户上传并发布,本平台仅提供信息存储服务。

Notice: The content above (including the pictures and videos if any) is uploaded and posted by a user of NetEase Hao, which is a social media platform and only provides information storage services.

相关推荐
热点推荐
10秒!美国发明革命性人体三维全景成像!

10秒!美国发明革命性人体三维全景成像!

徐德文科学频道
2026-01-19 18:56:23
徐帆带徐朵参加活动,3次想牵女儿手均被拒绝!徐朵发文回应质疑

徐帆带徐朵参加活动,3次想牵女儿手均被拒绝!徐朵发文回应质疑

露珠聊影视
2026-01-19 10:32:13
一声叹息:U23国足晋级四强,但2位19岁神童落寞,王钰栋最憋屈

一声叹息:U23国足晋级四强,但2位19岁神童落寞,王钰栋最憋屈

国足风云
2026-01-19 08:44:18
“儿子都腺样体了,还吃烤肠呢?”一份糊弄早餐暴露家长的低认知

“儿子都腺样体了,还吃烤肠呢?”一份糊弄早餐暴露家长的低认知

妍妍教育日记
2025-12-26 17:18:26
冬天控制体重的5个方法:坚持“早上2个坚持,晚上3个不”

冬天控制体重的5个方法:坚持“早上2个坚持,晚上3个不”

增肌减脂
2026-01-19 18:33:30
事发飞往上海航班!男乘客突然意识模糊!居然冲出三名专家,医生精准判断

事发飞往上海航班!男乘客突然意识模糊!居然冲出三名专家,医生精准判断

环球网资讯
2026-01-18 08:09:40
Lisa金球奖庆功派对造型惹争议,恢复铁刘海妆造的她,却穿透视装

Lisa金球奖庆功派对造型惹争议,恢复铁刘海妆造的她,却穿透视装

小老头奇闻
2026-01-16 15:18:15
继续停课一天!郑州市内多区发布停课通知

继续停课一天!郑州市内多区发布停课通知

大象新闻
2026-01-19 17:41:17
小孩为什么从不主动开口要钱? 网友: 知道孩子要用钱就该定时打钱

小孩为什么从不主动开口要钱? 网友: 知道孩子要用钱就该定时打钱

另子维爱读史
2026-01-18 21:26:18
你发现了吗?iPhone 灵动岛开始变成广告岛了

你发现了吗?iPhone 灵动岛开始变成广告岛了

XCiOS俱乐部
2026-01-18 14:52:06
日本AV女演员生存实录:日工作14小时,她们最后都嫁给了谁?

日本AV女演员生存实录:日工作14小时,她们最后都嫁给了谁?

蔚蓝的珊瑚海
2025-12-30 23:01:24
董璇前夫高云翔深夜晒淋雪照!边走边笑,对比张维伊差太多

董璇前夫高云翔深夜晒淋雪照!边走边笑,对比张维伊差太多

一盅情怀
2026-01-19 14:10:47
突发!腿筋拉伤,无限休战!西部又要大乱

突发!腿筋拉伤,无限休战!西部又要大乱

余憁搞笑段子
2026-01-19 11:21:41
烧烤摊上20元一串的“油边”,到底是动物的啥部位?

烧烤摊上20元一串的“油边”,到底是动物的啥部位?

大象新闻
2026-01-13 18:57:16
特朗普再提俄罗斯威胁论

特朗普再提俄罗斯威胁论

财联社
2026-01-19 12:24:09
1966年,一位农民因一头驴,写信给中央表示 “希望老政委帮忙”,邓小平是如何回应并处理的?

1966年,一位农民因一头驴,写信给中央表示 “希望老政委帮忙”,邓小平是如何回应并处理的?

老杉说历史
2026-01-16 20:00:09
19号收评:三大指数涨跌不一,所有人都注意,大盘后市或将这样走

19号收评:三大指数涨跌不一,所有人都注意,大盘后市或将这样走

春江财富
2026-01-19 15:21:49
“一周干完一年活”!Claude Code让工程师集体“上瘾”:写代码、操控电脑、整理文件夹,背后却埋安全暗雷

“一周干完一年活”!Claude Code让工程师集体“上瘾”:写代码、操控电脑、整理文件夹,背后却埋安全暗雷

每日经济新闻
2026-01-19 17:19:06
德里赫特下周复出,恐缺席曼联战阿森纳!谢什科无缘德比原因曝光

德里赫特下周复出,恐缺席曼联战阿森纳!谢什科无缘德比原因曝光

罗米的曼联博客
2026-01-19 11:13:13
人一过60,永远不要在熟人面前,说以下几句话,谁说谁后悔,切记

人一过60,永远不要在熟人面前,说以下几句话,谁说谁后悔,切记

枫红染山径
2026-01-02 14:59:55
2026-01-19 19:32:49
Python猫 incentive-icons
Python猫
人生苦短,我用Python。博客:https://pythoncat.top
729文章数 8121关注度
往期回顾 全部

科技要闻

这一仗必须赢!马斯克死磕芯片"9个月一更"

头条要闻

24岁唇腭裂患者接受了免费手术:嫣然基金会做了好事

头条要闻

24岁唇腭裂患者接受了免费手术:嫣然基金会做了好事

体育要闻

错失英超冠军奖牌,他却在德甲成为传奇

娱乐要闻

离婚三年,孙怡董子健首次公开互动

财经要闻

公章争夺 家族反目 双星为何从顶端跌落?

汽车要闻

徐军:冲击百万销量,零跑一直很清醒

态度原创

家居
本地
教育
房产
公开课

家居要闻

隽永之章 清雅无尘

本地新闻

云游内蒙|黄沙与碧波撞色,乌海天生会“混搭”

教育要闻

2026年大学俄语四六级1分钟速通攻略!

房产要闻

中旅・三亚蓝湾发布会揭秘自贸港好房子高阶形态

公开课

李玫瑾:为什么性格比能力更重要?

无障碍浏览 进入关怀版