Pycharm远程调试原理及具体配置详解


Posted in Python onAugust 08, 2019

前言

工作中使用Pycharm作为python开发的IDE,作为专业的python集成开发环境,其功能之强大令人折服。开发过程中Debug是必不可少的。平时经常使用Pycharm的remote debug功能,非常好用。但是刚开始的时候并不了解该过程的原理,只是按部就班的配置。于是抽空了解了一下相关知识,期待能够了解其原理,今后能够在需要的时候自己独立的配置调试环境。本文将以浅显易懂的方式讲解一下相关过程。

1.应用环境

常见的IDE基本都具有Local Debugger功能。一般只需要简单的配置,直接加断点并使用Debug方式运行即可使用断点调试。这是对于本地调试开发而言。如果项目已经完成并上线部署到服务端,或者是本地需要在IDE之外单独配置并启动程序,那么显然不能使用本地调试。如果能够配合日志并使用断点定位分析问题,将会事半功倍。那么如何使用本地安装的Pycharm远程调试程序?

2.远程调试原理

如果程序部署在远端,要在本地获取程序运行状态并进行断点调试,必然需要连接到程序并进行通讯。利用Pycharm进行远程调试过程中,Pycharm充当服务器的角色。

首先,对Pycharm Run/Debug Configures进行配置,指定Pycharm安装端的一些属性,比如Pycharm所在主机的IP地址和端口号等。

然后,启动Pycharm的远程调试。这时Pycharm处于监听状态,等待独立于IDE之外运行的程序的连接。

其次,在远端程序刚启动时,需要根据Pycharm Debug Configures中的配置信息,连接到Pycharm。

最后,连接成功之后,当远程客户端运行到本地Pycharm中设置的断点处时,便会在断点处暂停程序的执行,而在本地Pycharm命中断点处能够看到远端程序当前运行的状态和调用栈等信息并进行下一步跟踪和逐步调试。

本地Pycharm中调试的源代码工程应和远端运行的程序源代码保持一致。Pycharm中Remote Debug Configure的配置要保证能够被远程连接。

由于远程客户端使用Pycharm提供的pydevd模块连接到本地的Pycharm remote Debug,两者通讯链接均遵循Pycharm自定义的协议。因此我们不必关心Pycharm设置断点和远程客户端命中断点过程中两端具体的实现和处理过程,只要保证我们的Debug Configure有效即可。然后在需要的地方通过断点暂定程序,分析当前程序状态找出问题所在。

3.Pycharm具体配置

下面是当前的操作环境,原理和步骤都是想通的,可根据实际情况进行配置配即可。

当前环境:

  • Win7
  • Python 2.7.12
  • PyCharm Professional 2017.1.4 (community版本 好像没有remote debug 功能)

本地完成Pycharm的安装,在安装目录找到debug-eggs文件夹,里面有两个文件:

  • pycharm-debug.egg
  • pycharm-debug-py3k.egg

分别对应本地python解释器为python2和python3的情况。

解压pycharm-debug.egg文件,得到的文件夹pycharm-debug中包含的是remote debug相关的模块。

远程客户端便是通过该文件夹中pydevd文件的settrace方法连接到指定的debug server的。

客户端配置:

为方便起见,我们将客户端也放置到本地。(远端的只需要将下面的localhost改为Pycharm所在端的IP即可)

工程中添加刚才解压得到的远程调试模块:

./pycharm-debug

下面还需要封装一个连接到Remote Debug Server的文件

./PycharmRemoteDebug.py

import sys
sys.path.append('./Pycharm_debug')
import pydevd
if __name__ != '__main__':
 pydevd.settrace('localhost', port=23456, stdoutToServer=True, stderrToServer=True, suspend=False)

当前待调试程序:./Main.py

# -*- coding:utf-8 -*-
import PycharmRemoteDebug

class Singleton(object):
 _INSTANCE = {}
 def __init__(self, cls):
  self.cls = cls

 def __call__(self, *args, **kwargs):
  instance = self._INSTANCE.get(self.cls, None)
  if not instance:
   instance = self.cls(*args, **kwargs)
   self._INSTANCE[self.cls] = instance
  return instance

 def __getattr__(self, key):
  return getattr(self.cls, key, None)
@Singleton
class MyClass(object):
 def __init__(self):
  self.init_attr = "init_attr"
 def __getattr__(self, key):
  return self.__dict__.get(key, 1212)
if __name__ == "__main__":
 mcls1 = MyClass()
 mcls2 = MyClass()
 print mcls1 is mcls2

注意调试模块pycharm-debug和链接文件PycharmRemoteDebug.py可以放置到任意的路径和位置,只需要调整模块引用的路径即可。

当前工程目录:

Pycharm远程调试原理及具体配置详解

Pycharm Remote Debug 配置:

打开Run/Debug Configures

Pycharm远程调试原理及具体配置详解

新建配置 Add New Configuration --> Python Remote Debug

Pycharm远程调试原理及具体配置详解

上面的名字可以自己随便命名,端口号可以随便改,只要可用即可。

上面截图绿色部分的标记也告诉了我们客户端连接Debug Server的步骤方法,注意第三步中使用的命令就是我们客户端配置中的PycharmRemoteDebug.py文件中的连接命令。

4.使用步骤

选择刚才新建的Debug模式 Remote_Debug,点击绿色甲壳虫Debug按钮:

Pycharm远程调试原理及具体配置详解

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中命中断点:

Pycharm远程调试原理及具体配置详解

Watch程序当前状态以及调用栈等信息:

Pycharm远程调试原理及具体配置详解

5.注意事项

  • 如果将程序部署到远端,那么需要Remote Debug配置中的localhost修改为安装Pycharm主机的IP地址,同时将PycharmRemoteDebug.py中的localhost改为同样的IP地址;
  • 使用Pycharm的Deployment功能映射远程和本地代码;
  • 断点命中时,客户端程序处于暂定状态;
  • 如果没有开启Remote Debug Server,运行客户端会卡住;
  • 不需要使用Remote Debug时一定不要在程序启动的时候import PycharmRemoteDebug模块;
  • 添加remote debug 配置文件后,注意区分启动本地和远程两种不同的调试模式;

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Python 相关文章推荐
纯Python开发的nosql数据库CodernityDB介绍和使用实例
Oct 23 Python
python获取文件后缀名及批量更新目录下文件后缀名的方法
Nov 11 Python
Python对象类型及其运算方法(详解)
Jul 05 Python
Python环境搭建之OpenCV的步骤方法
Oct 20 Python
Python_LDA实现方法详解
Oct 25 Python
Python分析学校四六级过关情况
Nov 22 Python
Python3.6通过自带的urllib通过get或post方法请求url的实例
May 10 Python
Python爬虫实现(伪)球迷速成
Jun 10 Python
Flask框架踩坑之ajax跨域请求实现
Feb 22 Python
深入浅析Python 中 is 语法带来的误解
May 07 Python
基于pycharm实现批量修改变量名
Jun 02 Python
python中HTMLParser模块知识点总结
Jan 25 Python
Python IDE Pycharm中的快捷键列表用法
Aug 08 #Python
python多线程与多进程及其区别详解
Aug 08 #Python
python PIL和CV对 图片的读取,显示,裁剪,保存实现方法
Aug 07 #Python
python实现对图片进行旋转,放缩,裁剪的功能
Aug 07 #Python
Django 实现前端图片压缩功能的方法
Aug 07 #Python
Django 后台获取文件列表 InMemoryUploadedFile的例子
Aug 07 #Python
python 直接赋值和copy的区别详解
Aug 07 #Python
You might like
关于时间计算的结总
2006/12/06 PHP
php上传文件常见问题总结
2015/02/03 PHP
Laravel学习教程之路由模块
2017/08/18 PHP
PHP实现浏览器格式化显示XML的方法示例
2019/01/22 PHP
改善你的jQuery的25个步骤 千倍级效率提升
2010/02/11 Javascript
高效的表格行背景隔行变色及选定高亮的JS代码
2010/12/04 Javascript
js工具方法弹出蒙版
2013/05/08 Javascript
ext中store.load跟store.reload的区别示例介绍
2014/06/17 Javascript
JavaScript声明变量时为什么要加var关键字
2014/09/29 Javascript
快速学习jQuery插件 jquery.validate.js表单验证插件使用方法
2015/12/01 Javascript
JavaScript来实现打开链接页面的简单实例
2016/06/02 Javascript
JavaScript探测CSS动画是否已经完成的方法
2016/08/30 Javascript
js学习笔记之事件处理模型
2016/10/31 Javascript
解决Layui选择全部,换页checkbox复选框重新勾选的问题方法
2018/08/14 Javascript
在vue和element-ui的table中实现分页复选功能
2019/12/04 Javascript
BootStrap前端框架使用方法详解
2020/02/26 Javascript
Javascript作用域和作用域链原理解析
2020/03/03 Javascript
JS脚本实现定时到网站上签到/签退功能
2020/04/22 Javascript
利用Python的装饰器解决Bottle框架中用户验证问题
2015/04/24 Python
Python使用微信SDK实现的微信支付功能示例
2017/06/30 Python
Python编程之字符串模板(Template)用法实例分析
2017/07/22 Python
Python使用requests及BeautifulSoup构建爬虫实例代码
2018/01/24 Python
python读取文本中数据并转化为DataFrame的实例
2018/04/10 Python
python生成带有表格的图片实例
2019/02/03 Python
Django RBAC权限管理设计过程详解
2019/08/06 Python
Python jieba库用法及实例解析
2019/11/04 Python
在python中修改.properties文件的操作
2020/04/08 Python
python 写一个水果忍者游戏
2021/01/13 Python
HTML5 canvas实现的静态循环滚动播放弹幕
2021/01/05 HTML / CSS
北承题目(C++)
2012/05/16 面试题
土木工程应届生求职信
2013/10/31 职场文书
新闻专业个人求职信
2013/12/19 职场文书
童装店创业计划书
2014/01/09 职场文书
施工安全承诺书
2014/05/22 职场文书
社团活动总结模板
2014/06/30 职场文书
利用Sharding-Jdbc进行分库分表的操作代码
2022/01/22 Java/Android