Python中,如何使用 IPython 調試(debug)程式

来源:https://www.cnblogs.com/jiaoran/archive/2022/05/08/16245921.html
-Advertisement-
Play Games

介紹瞭如何在程式代碼中嵌入IPython用於調試,並分析了優點與不足 ...


關於IPython使用的入門文章,主要介紹瞭如何在程式代碼中嵌入ipython用於調試,並分析了優點與不足。

在 Python 中編程時,我會花費大量時間使用 IPython 及其強大的互動式提示,不僅用於一些一次性計算,還用於大量實際編程和調試。我特別將它用於一些探索性的編程,比如對一些不熟悉的 API,或者想知道程式在代碼中特定位置的運行狀態。

我不確定這種IPython調試的方法有多普遍,但我很少聽到其他人談論它,所以我認為它值得分享。

安裝

使用前,需要將 IPython 安裝到您當前的 virtualenv 中:

pip install ipython

使用方法

基本上有兩種方法可以打開 IPython 提示符。

第一種是直接從終端運行它:

$ ipython
Python 3.9.5 (default, Jul  1 2021, 11:45:58)
Type 'copyright', 'credits' or 'license' for more information
IPython 8.3.0 -- An enhanced Interactive Python. Type '?' for help.

In [1]:

在 Django 項目中,如果您安裝了 IPython,也可以使用 ./manage.py shell,好處是它會為幫您正確初始化 Django。

如果您想探索編寫一些“頂級”代碼,例如,在尚未創建入口點的情況下,編寫一個新的功能,那麼這種方法很管用。然而,我寫的大部分代碼都不是這樣的。大多數時候,我發現自己需要寫代碼時,已經想好10層的函數調用了——比如:

  • 我正在一個Django應用程式中編寫一些視圖代碼,其中有一個請求對象--如果你在IPython提示符下從頭開始,你不可能輕易重新創建這個對象。
  • 或者,模型層代碼,比如 save() 方法內部的代碼,該方法本身正在被您尚未編寫的其他代碼調用,比如Django admin或某個信號。
  • 或者,在一個測試中,設置代碼已經創建了一大堆在打開IPython時不可用的東西。

對於這些情況,我使用第二種方法:

  • 找到我想要修改、探索或調試的代碼。這通常是我自己的代碼,但也可能是第三方庫。我一直習慣在 virtualenv 中工作,所以即使使用第三方庫,在我的編輯器中“go to definition”也會直接將我帶到代碼的可寫副本的定義區(除了不是用 Python 編寫的代碼)。
  • 插入 IPython 提示的代碼並保存文件:
import IPython; IPython.embed()

我將此綁定到編輯器中的一個功能鍵。
因此,如果它是Django視圖,那麼代碼最終可能會是這樣:

def contact_us(request):
    if request.method == "POST":
        form = ContactUsForm_class(request.POST)
        if form.is_valid():
            import IPython; IPython.embed()

        # …
  • 以適當的方式觸發代碼。對於上述情況,首先需要在終端中運行 Django 伺服器,然後打開網頁,填寫表單並按下提交。對於測試,它將從終端運行特定的測試。對於命令行應用程式,它將直接運行應用程式。
  • 在終端中,我會發現自己現在已經在 IPython REPL 中,我可以繼續:
    • 想出我需要寫什麼代碼
    • 或者調試我感到困惑的代碼

請註意,您可以在此 REPL 中編寫和編輯多行代碼——它不像編輯器那麼舒服,但沒關係,並且具有良好的歷史記錄支持。關於 IPython 及其更多特性,你可以在官方 文檔 中瞭解它。

對於那些有其他語言背景的人來說,可能還值得指出的是,Python REPL 與普通 Python 並沒有什麼不同。你可以在普通 Python 中做的所有事情,比如定義函數和類,都可以在 REPL 中進行。

調試結束後,我可以將任何有用的片段從 REPL 複製回我的真實代碼中,使用歷史記錄來查看我曾經輸入的內容。

優點

這種方法的優點是:

  1. 當您實際擁有一個對象時,您可以更輕鬆地探索API和對象(APIs and objects),而不是閱讀關於對象的文檔,或者編輯器的自動完成工具推斷對象應該具有的內容。例如,Django的HttpRequest上有哪些屬性和方法?你不必確保你有正確的類型註釋,並且希望它們是完整的,或者假設值是什麼——你已經有了對象,你可以檢查它,用廣泛的合適的製表符自動補全完成。你可以調用函數,看看它們是怎麼做的。
    例如,Django的請求對象通常有一個用戶(user)屬性,該屬性不屬於HttpRequest定義的一部分,因為它是在以後添加的。但它在REPL中是可見的。
  2. 您可以直接探索程式的整體狀態。這對於探索性編程和調試來說都是一個巨大的優勢。
    對於調試,pdb 和類似的調試工具和環境通常會為您提供“the state of the system”,並且它們更擅長單步執行多層代碼。但我經常發現 IPython 提示的功能和舒適性對於探索和尋找解決方案要好得多。

這種環境的感覺並不像Lisp中REPL驅動的編程那樣流暢,但我仍然覺得它非常有趣和高效。與許多其他方法相比,比如迭代代碼,然後進行手動或自動測試,它將反饋迴圈的延遲從幾秒或幾分鐘減少到幾毫秒,這是巨大的效率提升。

提示和不足

  • IPython 有很多很酷的特性可以在 REPL 環境中幫助你,比如 %autoreload 和許多其他很酷的魔法。你應該花時間去瞭解他們!
  • 在多線程(或多進程)環境中,IPython 提示表現不是很好。如果可能的話,關閉多線程,或者確保你沒有遇到那個問題。
  • 如果您確實在終端中搞砸了,您可能需要手動找到要殺死的進程併在終端中進行重置。
  • 使用 Django 開發伺服器:
    • 它預設是多線程的,所以要麼確保你不會多次點擊視圖代碼,要麼使用 --nothreading
    • 當心自動重新載入,如果你在啟動時仍然處於 IPython 提示符中,它會搞砸你。要麼使用 --noreload 要麼確保在執行任何會觸發重新載入的操作之前乾凈地退出 IPython。
  • 當心捕獲標準輸入/輸出的環境,這會破壞這種功能。
  • pytest 預設捕獲標準輸入並破壞一些事物。您可以使用 -s 將其關閉。此外,如果您使用的是 pytest-xdist,您應該記得使用 -n0 來關閉多個進程。
  • 使用 IPython.embed() 時,由於 Python 的限制,存在一個煩人的錯誤,涉及閉包和未定義的名稱。它經常在使用生成器表達式時出現,但在其他時候也是如此。它通常可以通過以下方式解決:
globals().update(locals())

參考鏈接


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • # JavaScript經典案例練習 P1:支付10秒倒計時 效果: 功能: 點擊立即支付彈出確認款 確認付款後跳轉到支付成功頁面並開始計時 計時結束或者點擊立即返回按鈕返迴首頁 知識點: window.confirm() 用於顯示一個帶有指定消息和確認及取消按鈕的對話框,如果訪問者點擊‘確認’, ...
  • ${}字元串可以用於字元串拼接,一般用於模糊查詢中(因為有sql註入的風險,很少用) ...
  • 在開發過程中,不知道有沒有這樣的經歷,項目實際讀取的配置信息有時候總是與預期不符,今天就來研究下 SpringBoot 讀取配置文件順序。 一、SpringBoot 配置文件載入優先順序 SpringBoot官方文檔說明瞭載入的順序如下,越靠前優先順序越高。 Spring Boot uses a ver ...
  • 前言 嗨嘍!大家好呀 第三方模塊: requests >>> pip install requests 模塊安裝問題: 如果安裝python第三方模塊: win + R 輸入 cmd 點擊確定, 輸入安裝命令 pip install 模塊名 (pip install requests) 回車 在py ...
  • 一、Matplotlib 博文來源:https://www.runoob.com/matplotlib/matplotlib-tutorial.html Matplotlib 是 Python 的繪圖庫,它能讓使用者很輕鬆地將數據圖形化,並且提供多樣化的輸出格式。 Matplotlib 可以用來繪製 ...
  • 在系統開發的過程中,必然存在耗時極高的動作,是基於請求響應模式無法解決的問題,通常會採用解耦的思維,並基於非同步或者事件驅動的方式去調度整個流程的完整執行。 ...
  • 一個工作了2年的粉絲,私信了一個比較簡單的問題。 說: “Spring中事務的傳播行為有哪些?” 他說他能記得一些,但是在項目中基本上不需要配置,所以一下就忘記了。 結果導致面試被拒絕,有點遺憾! ok,關於這個問題,看看普通人和高手的回答。 普通人: 嗯。。。。。。。。 高手: 對於這個問題,需要 ...
  • 函數的定義和使用 def test(x): # x代表形參 ''' 2*x+1 :param x:整形數字 :return:返回計算結果 ''' y = 2*x+1 return y p = test(3) # test()表示運行名為test函數,3代表實參,給x進行賦值 print(p) 函數的 ...
一周排行
    -Advertisement-
    Play Games
  • Dapr Outbox 是1.12中的功能。 本文只介紹Dapr Outbox 執行流程,Dapr Outbox基本用法請閱讀官方文檔 。本文中appID=order-processor,topic=orders 本文前提知識:熟悉Dapr狀態管理、Dapr發佈訂閱和Outbox 模式。 Outbo ...
  • 引言 在前幾章我們深度講解了單元測試和集成測試的基礎知識,這一章我們來講解一下代碼覆蓋率,代碼覆蓋率是單元測試運行的度量值,覆蓋率通常以百分比表示,用於衡量代碼被測試覆蓋的程度,幫助開發人員評估測試用例的質量和代碼的健壯性。常見的覆蓋率包括語句覆蓋率(Line Coverage)、分支覆蓋率(Bra ...
  • 前言 本文介紹瞭如何使用S7.NET庫實現對西門子PLC DB塊數據的讀寫,記錄了使用電腦模擬,模擬PLC,自至完成測試的詳細流程,並重點介紹了在這個過程中的易錯點,供參考。 用到的軟體: 1.Windows環境下鏈路層網路訪問的行業標準工具(WinPcap_4_1_3.exe)下載鏈接:http ...
  • 從依賴倒置原則(Dependency Inversion Principle, DIP)到控制反轉(Inversion of Control, IoC)再到依賴註入(Dependency Injection, DI)的演進過程,我們可以理解為一種逐步抽象和解耦的設計思想。這種思想在C#等面向對象的編 ...
  • 關於Python中的私有屬性和私有方法 Python對於類的成員沒有嚴格的訪問控制限制,這與其他面相對對象語言有區別。關於私有屬性和私有方法,有如下要點: 1、通常我們約定,兩個下劃線開頭的屬性是私有的(private)。其他為公共的(public); 2、類內部可以訪問私有屬性(方法); 3、類外 ...
  • C++ 訪問說明符 訪問說明符是 C++ 中控制類成員(屬性和方法)可訪問性的關鍵字。它們用於封裝類數據並保護其免受意外修改或濫用。 三種訪問說明符: public:允許從類外部的任何地方訪問成員。 private:僅允許在類內部訪問成員。 protected:允許在類內部及其派生類中訪問成員。 示 ...
  • 寫這個隨筆說一下C++的static_cast和dynamic_cast用在子類與父類的指針轉換時的一些事宜。首先,【static_cast,dynamic_cast】【父類指針,子類指針】,兩兩一組,共有4種組合:用 static_cast 父類轉子類、用 static_cast 子類轉父類、使用 ...
  • /******************************************************************************************************** * * * 設計雙向鏈表的介面 * * * * Copyright (c) 2023-2 ...
  • 相信接觸過spring做開發的小伙伴們一定使用過@ComponentScan註解 @ComponentScan("com.wangm.lifecycle") public class AppConfig { } @ComponentScan指定basePackage,將包下的類按照一定規則註冊成Be ...
  • 操作系統 :CentOS 7.6_x64 opensips版本: 2.4.9 python版本:2.7.5 python作為腳本語言,使用起來很方便,查了下opensips的文檔,支持使用python腳本寫邏輯代碼。今天整理下CentOS7環境下opensips2.4.9的python模塊筆記及使用 ...