Python 必備 debug 神器:pdb

個人小程序【編程面試題庫】

目錄

  • 叨逼叨
  • 使用介紹
  • 舉個簡單的栗子

叨逼叨

首先,介紹一下 pdb 調試,pdb 是 python 的一個內置模塊,用於命令行來調試 Python 代碼。或許你會說,如今用 Pycharm 等編輯器來調試代碼很方便,爲啥要用命令行呢?這個問題,我曾經也這麼想,直到有一次,代碼必需要在 Linux 系統上跑(如今 Pycharm 也能夠遠程調試代碼了,咱們先不說這個)html

使用介紹

如何添加斷點?

說到 debug,確定是要添加斷點的,這裏有兩種方式添加斷點:python

  • 在想要斷點代碼後添加 一行
pdb.set_trace()
複製代碼

如果使用這種方式,直接運行 Python 文件便可進入斷點調試。面試

  • 用命令行來添加斷點
b line_number(代碼行數)
複製代碼

如果使用這種方式,須要 python -m pdb xxx.py 來啓動斷點調試。express

經常使用命令

先簡單介紹一下使用命令,這裏不用記住,等用到的時候回來查就行。編程

  • 1 進入命令行Debug模式,python -m pdb xxx.py小程序

  • 2 h:(help)幫助bash

  • 3 w:(where)打印當前執行堆棧app

  • 4 d:(down)執行跳轉到在當前堆棧的深一層(我的沒以爲有什麼用處)scrapy

  • 5 u:(up)執行跳轉到當前堆棧的上一層編輯器

  • 6 b:(break)添加斷點

    b 列出當前全部斷點,和斷點執行到統計次數
    
           b line_no:當前腳本的line_no行添加斷點
    
           b filename:line_no:腳本filename的line_no行添加斷點
    
           b function:在函數function的第一條可執行語句處添加斷點
    複製代碼
  • 7 tbreak:(temporary break)臨時斷點

    在第一次執行到這個斷點以後,就自動刪除這個斷點,用法和b同樣
    複製代碼
  • 8 cl:(clear)清除斷點

    cl 清除全部斷點
    
          cl bpnumber1 bpnumber2... 清除斷點號爲bpnumber1,bpnumber2...的斷點
    
          cl lineno 清除當前腳本lineno行的斷點
    
          cl filename:line_no 清除腳本filename的line_no行的斷點
    複製代碼
  • 9 disable:停用斷點,參數爲bpnumber,和cl的區別是,斷點依然存在,只是不啓用

  • 10 enable:激活斷點,參數爲bpnumber

  • 11 s:(step)執行下一條命令

    若是本句是函數調用,則s會執行到函數的第一句
    複製代碼
  • 12 n:(next)執行下一條語句

    若是本句是函數調用,則執行函數,接着執行當前執行語句的下一條。
    複製代碼
  • 13 r:(return)執行當前運行函數到結束

  • 14 c:(continue)繼續執行,直到遇到下一條斷點

  • 15 l:(list)列出源碼

    l 列出當前執行語句周圍11條代碼
    
           l first 列出first行周圍11條代碼
    
           l first second 列出first--second範圍的代碼,若是second<first,second將被解析爲行數
    複製代碼
  • 16 a:(args)列出當前執行函數的函數

  • 17 p expression:(print)輸出expression的值

  • 18 pp expression:好看一點的p expression

  • 19 run:從新啓動debug,至關於restart

  • 20 q:(quit)退出debug

  • 21 j lineno:(jump)設置下條執行的語句函數

    只能在堆棧的最底層跳轉,向後從新執行,向前可直接執行到行號
    複製代碼
  • 22)unt:(until)執行到下一行(跳出循環),或者當前堆棧結束

  • 23)condition bpnumber conditon,給斷點設置條件,當參數condition返回True的時候bpnumber斷點有效,不然bpnumber斷點無效

舉個簡單的栗子

爲了驗證一下 pdb 的用法,我寫了個簡單的 Python 代碼,以下:

__author__ = 'zone'
__gzh__ = '公號: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()
複製代碼

運行實例:(這裏爲了方便你們閱讀,我添加了中文註釋,實際運行時不會有註釋的)

D:\work\venv\Scripts\python.exe D:/work_test/test/pdb_test/pdb_test.py
> d:\work_test\test\pdb_test\pdb_test.py(11)start_url()
-> for url in urls:


(Pdb) n  註釋:n(next)執行下一步
> d:\work_test\test\pdb_test\pdb_test.py(12)start_url()
-> print(url)


(Pdb) l  註釋: l(list)列出當前代碼
  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),繼續執行,知道遇到下一個斷點
http://www.zone7.cn
http://www.zone7.cn
http://www.zone7.cn
http://www.zone7.cn
> d:\work_test\test\pdb_test\pdb_test.py(17)parse()
-> for url in self.urls:


(Pdb) n  註釋:n(next)執行下一步
> d:\work_test\test\pdb_test\pdb_test.py(18)parse()
-> result = self.request_something(url)


(Pdb) l 註釋: l(list)列出當前代碼
 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)這裏是進入 request_something() 函數的意思 --Call-- > d:\work_test\test\pdb_test\pdb_test.py(20)request_something() -> def request_something(self, url): (Pdb) n 註釋:n(next)執行下一步 > d:\work_test\test\pdb_test\pdb_test.py(21)request_something() -> print('requesting...') (Pdb) l 註釋: l(list)列出當前代碼 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 變量的數據
'http://www.zone7.cn'


(Pdb) n  註釋:n(next)執行下一步
requesting...
> d:\work_test\test\pdb_test\pdb_test.py(31)request_something()
-> </html>''' (Pdb) p data 註釋:p(print)打印出指定變量的數據,這裏因爲賦值還沒完成,因此報錯 *** NameError: name 'data' is not defined (Pdb) n 註釋:n(next)執行下一步 > d:\work_test\test\pdb_test\pdb_test.py(32)request_something() -> return data (Pdb) p data 註釋:p(print)打印出指定變量的數據 '<!DOCTYPE html>\n<html lang="en">\n<head>\n    <meta charset="UTF-8">\n    <title>Title</title>\n</head>\n<body>\n\n</body>\n</html>' (Pdb) q 註釋:q(quit)退出 複製代碼

總結

按照上面的例子一套下來,基本的用法就能夠學會了,關鍵仍是得本身多實踐,今天就寫到這,還想寫一篇關於性能調試的文章,不知道這兩天有沒有時間了。

相關文章
相關標籤/搜索