用C语言对Python进行扩展

python C 扩展代码
优点:

  • 可以添加额外的功能:Python的多线程模型受限于GIL锁,自身提供的多线程模型实际上只能同时运行一个线程,但可以通过C扩展解决这个问题~
  • 性能提升:可以用Python构建模型,然后找出性能瓶颈的部分,用C进行重构
  • 私密性提升:解释性语言都是通过源码执行,然而源码文件缺少私密性,虽然可以发布预编译过的pyc文件。但是仍没有足够的安全性,所以可以通过C扩展,将一些核心模块通过C语言编写,发布编译后的二进制文件

Note:以下是PythonC扩展的代码示例,关键部分加入代码注释

    1 #include <stdio.h>
    2 #include <stdlib.h>
    3 #include <string.h>
    4 #include "Python.h"
    5 # "Python.h" 引入python api 头文件,这样才可以调用Python的C api 对C源码 进行包装
    6 #define BUFSIZE 10
    7 # 定义递归函数 fac 计算阶乘
    8 int fac(int n){
    9     if(n < 2)
   10         return 1;
   11     return n * fac(n - 1);
   12 }
   13 # 包装函数的用处就是先把python的值传递给C
        # 然后再把C中函数的计算结果转换成Python对象返回给Python
        # 需要为所有想被Python环境访问到的函数都增加一个静态函数,返回类型为PyObject *
        # 函数名格式为模块名_函数名;
   14 static PyObject * Extest_fac(PyObject *self, PyObject *args)
   15 {
   16     int num;
        # PyArg_ParseTuple(args,"i",&num)
        # 第一个参数是被解析的参数变量
        # 第二个参数是一个字符串,告诉我们如何解析元组中的每一个字符
        # "i" 代表整形, "s"代表字符串类型,"O"则代表一个Python对象
        #  接下来的参数是你希望通过PyArg函数解析并且保存的元素
   17     if(!(PyArg_ParseTuple(args,"i",&num))){
   18         return NULL;
   19     }
         # Py_BuildValue()
         # 将C 的数据类型与Python 的数据类型进行相互转换
         # 这里的i 表示将c语言的int 型 转为python 的int 型。
   20     return (PyObject *)Py_BuildValue("i",fac(num));
   21 }
   22
   23 char *reverse(char *s){
   24     register char t;
   25     char *p = s;
   26     char *q = (s + strlen(s) - 1);
   27     while(p < q)
   28     {
   29         t = *p;
   30         *p++ = *q;
   31         *q-- = t;
   32     }
   33     return s;
   34 }
   35 static PyObject * Extest_reverse(PyObject *self, PyObject *args){
   36     char *orignal;
   37     if(!(PyArg_ParseTuple(args,"s",&orignal))){
   38         return NULL;
   39     }
            # 这里与上述代码描述类似, 
            # 区别是将C语言的字符型变量转为Python 的字符型变量
   40     return (PyObject *)Py_BuildValue("s", reverse(orignal));
   41 }
   42
   43
        # 我们已经创建了几个包装函数,需要将这些函数列出来,以便Python解释器
        # 能够导入并且调用这些函数
        # 每一个数组都包含一个函数信息,METH_VARARGS代表参数以tuple的形式传入
        # 最后一个数组放置两个NULL 值,代表声明结束
   57 static PyMethodDef
   58 ExtestMethods[] = {
   59     {"fac",Extest_fac,METH_VARARGS},
   60     {"reverse",Extest_reverse,METH_VARARGS},
   61     {NULL,NULL},
   62 };
   63
        # 模块初始化,这部分代码在模块被Python导入时进行调用
   64 void initExtest(){
   65     Py_InitModule("Extest",ExtestMethods);
   66 }

编译与测试
为了让你的新Python扩展可以被创建,你需要将这些扩展与Python库放在一起编译。python的distutils包被用来编译,安装和分发这些模块,扩展,和包。

  • 创建setup.py
    我们在安装Python第三方包时,很多情况下会用到 Python setup.py install这个命令。
    下面我们来了解下setup.py文件的内容。
    编译的最主要内容都由setup函数完成,你需要为每一个扩展创建一个Extension实例。
#!/usr/bin/env python
from distutils.core import setup, Extension
        MOD = "Extest"
        setup(name = MOD , ext_modules = [Extension(MOD, sources = ['Extest.c'])])
# Extension 第一个参数是扩展的名字
# 如果模块是包的一部分,还需要加".";
# 第二个参数是源码的代码文件列表
  • 通过运行setup.py来编译和链接你的代码
python setup.py build
  • 进行调试
from ctypes import *
import os 
# 编译后生成的*.so文件路径
extest = cdll.LoadLibrary(os.getwd() + 'Extest.so')
print extest.fac(4)
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 194,491评论 5 459
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 81,856评论 2 371
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 141,745评论 0 319
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 52,196评论 1 263
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 61,073评论 4 355
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 46,112评论 1 272
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 36,531评论 3 381
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 35,215评论 0 253
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 39,485评论 1 290
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 34,578评论 2 309
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 36,356评论 1 326
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 32,215评论 3 312
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 37,583评论 3 299
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 28,898评论 0 17
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 30,174评论 1 250
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 41,497评论 2 341
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 40,697评论 2 335

推荐阅读更多精彩内容