首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >Python内核源码解析与C/CPP-API拓展编程(一)PyObject

Python内核源码解析与C/CPP-API拓展编程(一)PyObject

作者头像
Pulsar-V
发布2019-04-18 14:29:04
2.1K0
发布2019-04-18 14:29:04
举报
文章被收录于专栏:Pulsar-VPulsar-V

下载Python源码

我这里使用Python3.5作为学习的源码

目录结构

|
--- Include: 包括Python提供的所有头文件, 可以用于c/c++扩展
--- Lib: Python的标准库, 全部都是用python写的
--- Modules: 包含了C语言编写的模块, 比如random, StringIO 等
--- Parser: 包含了python解释器中的scanner和parser部分,也就是词法分析和语法分析部分,一个类似yacc一样根据规则自动生成
--- Objects: 包含所有Python的内置对象,整数, list, dict等.也包含了运行时python需要的所有内部使用的对象的实现
--- Python: 包含了python解释器中Compiler和执行引擎部分,是python运行的核心所在
--- PCBuild:包含了vs工程文件

调试方法:

基于C++的调试对于已经到Python虚拟机中存储起来的字节码命令是无法被观察到的,我们只能把它们解析成AST才能看懂字节码在解释器内存中的状态,所以这里我们借用Python解释器里的C_API来输出我们的对象:

PyAPI_FUNC(int) PyObject_Print(PyObject *, FILE *, int);

# eg:
PyObject_Print(v, stdout, 0);

Python架构

Python的整体架构可分为3个模块

  1. 内建模块 Python提供的大量的模块、库以及用户自定义的模块,比如import math,math就是python的内建模块。
  2. Python的运行时环境,包括对象/类型系统(Object/Type structures)、内存分配器(Memory Allocator)和运行时状态信息(Current State of Python)。 对象/类型系统:包含Python中存在的各种内建对象,int、list、dict等,以及用户自定义的各种类型和对象。
  3. 内存分配器:负责Python中创建对象时,对内存的申请工作,实际上是Python运行时与C中malloc的一层接口。
  4. 运行时状态:维护了解释器在执行字节码时不同的状态(正常状态和异常状态)之间的切换,有穷状态机。
  5. Python解释器或称为虚拟机,包括Scanner词法分析器,Parser语法分析器 ,Compiler编译器,Code Evaluator虚拟机。
  6. Scanner:将输入的Python源代码或从命令输入的Python代码分割成一个个的token。
  7. Parser:在Scanner的分析结果上进行语法分析,建立抽象语法树(AST)。
  8. Compiler:根据建立的AST生成指令集合—-Python字节码(byte code)
  9. Code Evaluator:执行字节码。

PyObject

python玩家都知道,在python中,万物皆对象,python的源码中是通过下面的代码去定义每一个对象的。

object.h

typedef struct _object {
    _PyObject_HEAD_EXTRA
    Py_ssize_t ob_refcnt;
    struct _typeobject *ob_type;
} PyObject;

结构体中包含了Py_ssize_t、_typeobject两个成员,下面一个个来看它的成员的定义和意义。

_PyObject_HEAD_EXTRA

object.h

#ifdef Py_TRACE_REFS
/* Define pointers to support a doubly-linked list of all live heap objects. */
#define _PyObject_HEAD_EXTRA            \
    struct _object *_ob_next;           \
    struct _object *_ob_prev;

#define _PyObject_EXTRA_INIT 0, 0,

#else
#define _PyObject_HEAD_EXTRA
#define _PyObject_EXTRA_INIT
#endif
Py_ssize_t

往下检索可以看到定义

pyport.h

typedef Py_intptr_t     Py_ssize_t;

pyport.h

typedef intptr_t        Py_intptr_t;

vcstdint.h

//预编译判断操作系统类型
#ifdef _WIN64 // [如果是64位操作系统
   typedef __int64           intptr_t;
   typedef unsigned __int64  uintptr_t;
#else // _WIN64 ][32位操作系统
   typedef _W64 int               intptr_t;
   typedef _W64 unsigned int      uintptr_t;
#endif // _WIN64 ]

__w64是一个编译器相关的关键字, 意思是说这个类型使用64位兼容方式编译, 在编译64位程序时指针就被视为64位宽, 而不是32位. int也有可能会被视为64位. __int64 原形为

typedef int __w64 __int64

换句话说在64位系统中, 它是64位int整型, 32位系统就是int. _W64 是为了兼容64位系统存在的. 所以 Py_ssize_t 的本质就是 int类型的变量。 将上面的定义全部回带化简以后,代码变成

typedef struct _object {
    __int64 ob_refcnt;//int ob_refcnt
    struct _typeobject *ob_type;
} PyObject;
作用

表示变量引用次数, python的垃圾回收机制基于引用计数, 在python运行的过程中当某个对象引用计数减少到0时, 就可以将该变量从堆上删除,释放内存。我们可以检索到引用计数的处理定义。

#define Py_INCREF(op)   ((op)->ob_refcnt++)          //增加计数
#define Py_DECREF(op)      \                         //减少计数        
     if (--(op)->ob_refcnt != 0)    \
         ;        \
     else         \
         __Py_Dealloc((PyObject *)(op))

引用计数为0时,该对象生命周期结束,python释放内存。

  • 引用计数机制的优点:
  • 简单
  • 实时性:一旦没有引用,内存就直接释放了。不用像其他机制等到特定时机。实时性还带来一个好处:处理回收内存的时间分摊到了运行时。
  • 引用计数机制的缺点:
  • 维护引用计数消耗资源
  • 循环引用

好了到此为止,涉及python的垃圾回收(GC)后面专门会分一篇文章来讲解

_typeobject

表示了对象的类型信息, 诸如int, string, function,class等,这里做一个简述,具体细节留到下一篇python类型对象去写。

 (adsbygoogle = window.adsbygoogle || []).push({});
本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 下载Python源码
  • 目录结构
  • 调试方法:
  • Python架构
  • PyObject
    • _PyObject_HEAD_EXTRA
      • Py_ssize_t
        • 作用
      • _typeobject
      领券
      问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档