python的debug實(shí)用工具 pdb詳解
叨逼叨
首先,介紹一下 pdb 調(diào)試,pdb 是 python 的一個(gè)內(nèi)置模塊,用于命令行來(lái)調(diào)試 Python 代碼?;蛟S你會(huì)說(shuō),現(xiàn)在用 Pycharm 等編輯器來(lái)調(diào)試代碼很方便,為啥要用命令行呢?這個(gè)問(wèn)題,我曾經(jīng)也這么想,直到有一次,代碼必須要在 Linux 系統(tǒng)上跑(現(xiàn)在 Pycharm 也可以遠(yuǎn)程調(diào)試代碼了,今天先不說(shuō)這個(gè))
使用介紹
如何添加斷點(diǎn)?
說(shuō)到 debug,肯定是要添加斷點(diǎn)的,這里有兩種方式添加斷點(diǎn):
在想要斷點(diǎn)代碼后添加 一行
pdb.set_trace()
若是使用這種方式,直接運(yùn)行 Python 文件即可進(jìn)入斷點(diǎn)調(diào)試。
用命令行來(lái)添加斷點(diǎn)
b line_number(代碼行數(shù))
若是使用這種方式,需要 python -m pdb xxx.py 來(lái)啟動(dòng)斷點(diǎn)調(diào)試。
常用命令
先簡(jiǎn)單介紹一下使用命令,這里不用記住,等用到的時(shí)候回來(lái)查就行。
1 進(jìn)入命令行Debug模式,python -m pdb xxx.py
2 h: (help)幫助
3 w: (where)打印當(dāng)前執(zhí)行堆棧
4 d: (down)執(zhí)行跳轉(zhuǎn)到在當(dāng)前堆棧的深一層(個(gè)人沒覺得有什么用處)
5 u: (up)執(zhí)行跳轉(zhuǎn)到當(dāng)前堆棧的上一層
6 b: (break)添加斷點(diǎn)
- b 列出當(dāng)前所有斷點(diǎn),和斷點(diǎn)執(zhí)行到統(tǒng)計(jì)次數(shù)
- b line_no:當(dāng)前腳本的line_no行添加斷點(diǎn)
- b filename:line_no:腳本filename的line_no行添加斷點(diǎn)
- b function:在函數(shù)function的第一條可執(zhí)行語(yǔ)句處添加斷點(diǎn)
7 tbreak: (temporary break)臨時(shí)斷點(diǎn)
在第一次執(zhí)行到這個(gè)斷點(diǎn)之后,就自動(dòng)刪除這個(gè)斷點(diǎn),用法和b一樣
8 cl: (clear)清除斷點(diǎn)
- cl 清除所有斷點(diǎn)
- cl bpnumber1 bpnumber2... 清除斷點(diǎn)號(hào)為bpnumber1,bpnumber2...的斷點(diǎn)
- cl lineno 清除當(dāng)前腳本lineno行的斷點(diǎn)
- cl filename:line_no 清除腳本filename的line_no行的斷點(diǎn)
9 disable:停用斷點(diǎn),參數(shù)為bpnumber,和cl的區(qū)別是,斷點(diǎn)依然存在,只是不啟用
10 enable:激活斷點(diǎn),參數(shù)為bpnumber
11 s: (step)執(zhí)行下一條命令
如果本句是函數(shù)調(diào)用,則s會(huì)執(zhí)行到函數(shù)的第一句
12 n: (next)執(zhí)行下一條語(yǔ)句
如果本句是函數(shù)調(diào)用,則執(zhí)行函數(shù),接著執(zhí)行當(dāng)前執(zhí)行語(yǔ)句的下一條。
13 r: (return)執(zhí)行當(dāng)前運(yùn)行函數(shù)到結(jié)束
14 c: (continue)繼續(xù)執(zhí)行,直到遇到下一條斷點(diǎn)
15 l: (list)列出源碼
- l 列出當(dāng)前執(zhí)行語(yǔ)句周圍11條代碼
- l first 列出first行周圍11條代碼
- l first second 列出first--second范圍的代碼,如果second<first,second將被解析為行數(shù)
16 a: (args)列出當(dāng)前執(zhí)行函數(shù)的函數(shù)
17 p expression:(print)輸出expression的值
18 pp expression:好看一點(diǎn)的p expression
19 run:重新啟動(dòng)debug,相當(dāng)于restart
20 q:(quit)退出debug
21 j lineno:(jump)設(shè)置下條執(zhí)行的語(yǔ)句函數(shù)
只能在堆棧的最底層跳轉(zhuǎn),向后重新執(zhí)行,向前可直接執(zhí)行到行號(hào)
22)unt:(until)執(zhí)行到下一行(跳出循環(huán)),或者當(dāng)前堆棧結(jié)束
23)condition bpnumber conditon,給斷點(diǎn)設(shè)置條件,當(dāng)參數(shù)condition返回True的時(shí)候bpnumber斷點(diǎn)有效,否則bpnumber斷點(diǎn)無(wú)效
舉個(gè)簡(jiǎn)單的栗子
為了驗(yàn)證一下 pdb 的用法,我寫了個(gè)簡(jiǎn)單的 Python 代碼,如下:
__author__ = 'zone'
__gzh__ = '公號(hào):zone7'
import pdb
class MyScrapy:
urls = []
def start_url(self, urls):
pdb.set_trace()
for url in urls:
print(url)
self.urls.append(url)
def parse(self):
pdb.set_trace()
for url in self.urls:
result = self.request_something(url)
def request_something(self, url):
print('requesting...')
data = '''<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
</body>
</html>'''
return data
scrapy= MyScrapy()
scrapy.start_url(["http://www.zone7.cn", "http://www.zone7.cn", "http://www.zone7.cn", "http://www.zone7.cn", ])
scrapy.parse()
運(yùn)行實(shí)例:(這里為了方便大家閱讀,我添加了中文注釋,實(shí)際運(yùn)行時(shí)不會(huì)有注釋的)
D:workenvScriptspython.exe D:/work_test/test/pdb_test/pdb_test.py
> d:work_test estpdb_testpdb_test.py(11)start_url()
-> for url in urls:
(Pdb) n 注釋:n(next)執(zhí)行下一步
> d:work_test estpdb_testpdb_test.py(12)start_url()
-> print(url)
(Pdb) l 注釋: l(list)列出當(dāng)前代碼
7 urls = []
8
9 def start_url(self, urls):
10 pdb.set_trace()
11 for url in urls:
12 -> print(url)
13 self.urls.append(url)
14
15 def parse(self):
16 pdb.set_trace()
17 for url in self.urls:
(Pdb) c 注釋:c(continue),繼續(xù)執(zhí)行,知道遇到下一個(gè)斷點(diǎn)
http://www.zone7.cn
http://www.zone7.cn
http://www.zone7.cn
http://www.zone7.cn
> d:work_test estpdb_testpdb_test.py(17)parse()
-> for url in self.urls:
(Pdb) n 注釋:n(next)執(zhí)行下一步
> d:work_test estpdb_testpdb_test.py(18)parse()
-> result = self.request_something(url)
(Pdb) l 注釋: l(list)列出當(dāng)前代碼
13 self.urls.append(url)
14
15 def parse(self):
16 pdb.set_trace()
17 for url in self.urls:
18 -> result = self.request_something(url)
19
20 def request_something(self, url):
21 print('requesting...')
22 data = '''<!DOCTYPE html>
23 <html lang="en">
(Pdb) s 注釋: s(step)這里是進(jìn)入 request_something() 函數(shù)的意思
--Call--
> d:work_test estpdb_testpdb_test.py(20)request_something()
-> def request_something(self, url):
(Pdb) n 注釋:n(next)執(zhí)行下一步
> d:work_test estpdb_testpdb_test.py(21)request_something()
-> print('requesting...')
(Pdb) l 注釋: l(list)列出當(dāng)前代碼
16 pdb.set_trace()
17 for url in self.urls:
18 result = self.request_something(url)
19
20 def request_something(self, url):
21 -> print('requesting...')
22 data = '''<!DOCTYPE html>
23 <html lang="en">
24 <head>
25 <meta charset="UTF-8">
26 <title>Title</title>
(Pdb) p url 注釋:p(print)打印出 url 變量的數(shù)據(jù)
'http://www.zone7.cn'
(Pdb) n 注釋:n(next)執(zhí)行下一步
requesting...
> d:work_test estpdb_testpdb_test.py(31)request_something()
-> </html>'''
(Pdb) p data 注釋:p(print)打印出指定變量的數(shù)據(jù),這里由于賦值還沒完成,所以報(bào)錯(cuò)
*** NameError: name 'data' is not defined
(Pdb) n 注釋:n(next)執(zhí)行下一步
> d:work_test estpdb_testpdb_test.py(32)request_something()
-> return data
(Pdb) p data 注釋:p(print)打印出指定變量的數(shù)據(jù)
'<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>Title</title>
</head>
<body>
</body>
</html>'
(Pdb) q 注釋:q(quit)退出
總結(jié)
按照上面的例子一套下來(lái),基本的用法就可以學(xué)會(huì)了,關(guān)鍵還是得自己多實(shí)踐,動(dòng)手操練!
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Python基于wxPython和FFmpeg開發(fā)一個(gè)視頻標(biāo)簽工具
在當(dāng)今數(shù)字媒體時(shí)代,視頻內(nèi)容的管理和標(biāo)記變得越來(lái)越重要,無(wú)論是研究人員需要對(duì)實(shí)驗(yàn)視頻進(jìn)行時(shí)間點(diǎn)標(biāo)記,還是個(gè)人用戶希望對(duì)家庭視頻進(jìn)行分類整理,一個(gè)高效的視頻標(biāo)簽工具都是不可或缺的,本文將詳細(xì)分析一個(gè)基于Python、wxPython和FFmpeg開發(fā)的視頻標(biāo)簽工具2025-04-04
python中queue.Queue之task_done的用法
這篇文章主要介紹了python中queue.Queue之task_done的用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-02-02
解析pip安裝第三方庫(kù)但PyCharm中卻無(wú)法識(shí)別的問(wèn)題及PyCharm安裝第三方庫(kù)的方法教程
這篇文章主要介紹了解析pip安裝第三方庫(kù)但PyCharm中卻無(wú)法識(shí)別的問(wèn)題及PyCharm安裝第三方庫(kù)的方法教程,本文圖文并茂給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2020-03-03
Python開發(fā)之迭代器&生成器的實(shí)戰(zhàn)案例分享
在 Python 中,迭代器和生成器都是用來(lái)遍歷數(shù)據(jù)集合的工具,可以按需逐個(gè)生成或返回?cái)?shù)據(jù),從而避免一次性加載整個(gè)數(shù)據(jù)集合所帶來(lái)的性能問(wèn)題和內(nèi)存消耗問(wèn)題。本文主要和大家分享幾個(gè)貼近實(shí)際運(yùn)維開發(fā)工作中的場(chǎng)景案例,希望對(duì)大家有所幫助2023-04-04
Python趣味入門教程之循環(huán)語(yǔ)句while
這篇文章主要給大家介紹了關(guān)于Python趣味入門教程之循環(huán)語(yǔ)句while的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-08-08
Pyqt5實(shí)戰(zhàn)小案例之界面與邏輯分離的小計(jì)算器程序
網(wǎng)上很多PyQt5信號(hào)槽與界面分離的例子,但是真正開發(fā)起來(lái)很不方便,下面這篇文章主要給大家介紹了關(guān)于Pyqt5實(shí)戰(zhàn)小案例之界面與邏輯分離的小計(jì)算器程序,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-02-02
python中*args與**kwarsg及閉包和裝飾器的用法
這篇文章主要介紹了python中*args與**kwarsg及閉包和裝飾器的用法說(shuō)明,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07

