工做中使用Pycharm做爲python開發的IDE,做爲專業的python集成開發環境,其功能之強大使人折服。開發過程當中Debug是必不可少的。平時常用Pycharm的remote debug功能,很是好用。可是剛開始的時候並不瞭解該過程的原理,只是循序漸進的配置。因而抽空了解了一下相關知識,期待可以瞭解其原理,從此可以在須要的時候本身獨立的配置調試環境。本文將以淺顯易懂的方式講解一下相關過程。python
常見的IDE基本都具備Local Debugger功能。通常只須要簡單的配置,直接加斷點並使用Debug方式運行便可使用斷點調試。這是對於本地調試開發而言。若是項目已經完成並上線部署到服務端,或者是本地須要在IDE以外單獨配置並啓動程序,那麼顯然不能使用本地調試。若是可以配合日誌並使用斷點定位分析問題,將會事半功倍。那麼如何使用本地安裝的Pycharm遠程調試程序?服務器
若是程序部署在遠端,要在本地獲取程序運行狀態並進行斷點調試,必然須要鏈接到程序並進行通信。利用Pycharm進行遠程調試過程當中,Pycharm充當服務器的角色。app
首先,對Pycharm Run/Debug Configures進行配置,指定Pycharm安裝端的一些屬性,好比Pycharm所在主機的IP地址和端口號等。spa
而後,啓動Pycharm的遠程調試。這時Pycharm處於監聽狀態,等待獨立於IDE以外運行的程序的鏈接。命令行
其次,在遠端程序剛啓動時,須要根據Pycharm Debug Configures中的配置信息,鏈接到Pycharm。debug
最後,鏈接成功以後,當遠程客戶端運行到本地Pycharm中設置的斷點處時,便會在斷點處暫停程序的執行,而在本地Pycharm命中斷點處可以看到遠端程序當前運行的狀態和調用棧等信息並進行下一步跟蹤和逐步調試。3d
本地Pycharm中調試的源代碼工程應和遠端運行的程序源代碼保持一致。Pycharm中Remote Debug Configure的配置要保證可以被遠程鏈接。調試
因爲遠程客戶端使用Pycharm提供的pydevd模塊鏈接到本地的Pycharm remote Debug,二者通信連接均遵循Pycharm自定義的協議。所以咱們沒必要關心Pycharm設置斷點和遠程客戶端命中斷點過程當中兩端具體的實現和處理過程,只要保證咱們的Debug Configure有效便可。而後在須要的地方經過斷點暫定程序,分析當前程序狀態找出問題所在。日誌
下面是當前的操做環境,原理和步驟都是想通的,可根據實際狀況進行配置配便可。code
當前環境:
本地完成Pycharm的安裝,在安裝目錄找到debug-eggs文件夾,裏面有兩個文件:
分別對應本地python解釋器爲python2和python3的狀況。
解壓pycharm-debug.egg文件,獲得的文件夾pycharm-debug中包含的是remote debug相關的模塊。
遠程客戶端即是經過該文件夾中pydevd文件的settrace方法鏈接到指定的debug server的。
客戶端配置:
爲方便起見,咱們將客戶端也放置到本地。(遠端的只須要將下面的localhost改成Pycharm所在端的IP便可)
工程中添加剛纔解壓獲得的遠程調試模塊:
./pycharm-debug
下面還須要封裝一個鏈接到Remote Debug Server的文件
./PycharmRemoteDebug.py
1 import sys 2 sys.path.append('./Pycharm_debug') 3 import pydevd 4
5 if __name__ != '__main__': 6 pydevd.settrace('localhost', port=23456, stdoutToServer=True, stderrToServer=True, suspend=False)
當前待調試程序:./Main.py
1 # -*- coding:utf-8 -*-
2 import PycharmRemoteDebug 3
4 class Singleton(object): 5 _INSTANCE = {} 6
7 def __init__(self, cls): 8 self.cls = cls 9
10 def __call__(self, *args, **kwargs): 11 instance = self._INSTANCE.get(self.cls, None) 12 if not instance: 13 instance = self.cls(*args, **kwargs) 14 self._INSTANCE[self.cls] = instance 15 return instance 16
17 def __getattr__(self, key): 18 return getattr(self.cls, key, None) 19
20 @Singleton 21 class MyClass(object): 22 def __init__(self): 23 self.init_attr = "init_attr"
24 def __getattr__(self, key): 25 return self.__dict__.get(key, 1212) 26
27 if __name__ == "__main__": 28 mcls1 = MyClass() 29 mcls2 = MyClass() 30 print mcls1 is mcls2
注意調試模塊pycharm-debug和連接文件PycharmRemoteDebug.py能夠放置到任意的路徑和位置,只須要調整模塊引用的路徑便可。
當前工程目錄:
Pycharm Remote Debug 配置:
打開Run/Debug Configures
新建配置 Add New Configuration --> Python Remote Debug
上面的名字能夠本身隨便命名,端口號能夠隨便改,只要可用便可。
上面截圖綠色部分的標記也告訴了咱們客戶端鏈接Debug Server的步驟方法,注意第三步中使用的命令就是咱們客戶端配置中的PycharmRemoteDebug.py文件中的鏈接命令。
選擇剛纔新建的Debug模式 Remote_Debug,點擊綠色甲殼蟲Debug按鈕:
Debug Console 顯示以下信息,說明本地Debug Server已經開啓並在監聽狀態:
Starting debug server at port 23456 Use the following code to connect to the debugger: import pydevd pydevd.settrace('localhost', port=23456, stdoutToServer=True, stderrToServer=True, suspend=False) Waiting for process connection...
而後在Pycharm中設置斷點。
最後啓動客戶端(運行Main.py,並不是在IDE中,直接雙擊該文件或者使用命令行執行)
Pycharm中命中斷點:
Watch程序當前狀態以及調用棧等信息: