http://www.web008.net

Python的C拓展简介

2.3 Python调用

那生机勃勃部分就大概了,步向含有编写翻译好的 .pyd 文件夹,新建如下文件:

import calc

print(calc.add(12, 21))

这便是三个普通库,那样调用就OK了。

2.1 C模块封装

以总结五个数相加为例,采纳随机文件夹,新建如下C语言源码:

// 文件名 calc.c
#include <Python.h>

int add(int x, int y){ // C 函数
    return x + y;
}

static PyObject *calc_add(PyObject *self, PyObject *args){

    int x, y;
    // Python传入参数
    // "ii" 表示传入参数为2个int型参数,将其解析到x, y变量中
    if(!PyArg_ParseTuple(args, "ii", &x, &y))
        return NULL;
    return PyLong_FromLong(add(x, y));
}

// 模块的方法列表
static PyMethodDef CalcMethods[] = {
     {"add", calc_add, METH_VARARGS, "函数描述"},
     {NULL, NULL, 0, NULL}
};

// 模块
static struct PyModuleDef calcmodule = {
    PyModuleDef_HEAD_INIT,
    "calc", // 模块名
    NULL, // 模块文档
    -1,       /* size of per-interpreter state of the module,
                or -1 if the module keeps state in global variables. */
    CalcMethods
};

// 初始化
PyMODINIT_FUNC PyInit_calc(void)
{
    return PyModule_Create(&calcmodule);
}

其中,静态函数 calcadd 以python的C接口方式封装了add函数,命超级模特式`模块名函数名`

静态PyMethodDef列表 变量 CalcMethods 包蕴了该模块方法的叙说

静态struct PyModuleDef结构体 变量 calcmodule 定义了模块的描述

PyInit_calc 函数伊始化了模块,命名情势 PyInit_模块名

C 文件

/**构建返回值
Py_BuildValue("")                        None
Py_BuildValue("i", 123)                  123
Py_BuildValue("iii", 123, 456, 789)      (123, 456, 789)
Py_BuildValue("s", "hello")              'hello'
Py_BuildValue("y", "hello")              b'hello'
Py_BuildValue("ss", "hello", "world")    ('hello', 'world')
Py_BuildValue("s#", "hello", 4)          'hell'
Py_BuildValue("y#", "hello", 4)          b'hell'
Py_BuildValue("()")                      ()
Py_BuildValue("(i)", 123)                (123,)
Py_BuildValue("(ii)", 123, 456)          (123, 456)
Py_BuildValue("(i,i)", 123, 456)         (123, 456)
Py_BuildValue("[i,i]", 123, 456)         [123, 456]
Py_BuildValue("{s:i,s:i}", "abc", 123, "def", 456)    {'abc': 123, 'def': 456}
Py_BuildValue("((ii)(ii)) (ii)", 1, 2, 3, 4, 5, 6)          (((1, 2), (3, 4)), (5, 6))
**/

#include<Python.h>


static PyObject *value_commonArgs(PyObject *self, PyObject *args){
    // 传入普通参数,例如: s = value.com(1, 2.3, "Hello C")
    int x;
    double y;
    char *z;
    if(!PyArg_ParseTuple(args, "ids", &x, &y, &z))
        return NULL;
    printf("The args is %d and %f and %s .n", x, y, z);
    // 返回(x, y, z)的元组
    return Py_BuildValue("(i,d,s)",x, y, z);
}



static PyObject *value_tupleTest(PyObject *self, PyObject *args){
    // t = value.tut((1, 3), "Tuple")
    int x, y;
    char *z;
    if(!PyArg_ParseTuple(args, "(ii)s", &x, &y, &z))
        return NULL;
    printf("The args is (%d, %d), %s .n", x, y, z);

    // return ([1, 2], "hello")
    return Py_BuildValue("[i,i]s", x, y, z);
}



static PyObject *value_some(PyObject *self, PyObject *args){
    /* 可选参数,可能是下面几种, “|” 代表后面的参数可选
        c = value.som(1)
        value.som(1, 3)
        value.som(1, 2, "hello")
    */
    int x = 0, y = 0;
    char *z = NULL;
    if(!PyArg_ParseTuple(args, "i|is", &x, &y, &z))
        return NULL;
    printf("x is: %dn", x);
    printf("y is: %dn", y);
    if(z != NULL)printf("z is: %sn", z);
    return Py_BuildValue("[i,i,s]", x, y, z

1. 境况计划

假如果Linux只必要设置Python3.x + Python-dev。

Windows下多少复杂点,VS2017 + Python3.6.3

VS2017可用社区版,供给接纳设置的条件如下:

图片 1

图片 2

2. Hello World !

2.2 C源码编写翻译

在VS2017中得以平昔扭转 .dll 文件,然后改名称为 .pyd 就可在python程序中引进该模块了,不过,那不“清真”,正确的架势是写二个setup.py然后通过python调cl.exe编写翻译。

新建setup.py文件,内容如下:

# setup.py

from distutils.core import setup, Extension

module1 = Extension('calc',
                    sources=['calc.c'])

setup(name='calc_model',
      version='1.0',
      description='Hello ?',
      ext_modules=[module1]
)

接下来,从Windows的命令行(命令提醒符)下进入到那些文件夹下,推行:

python setup.py build

就能够产生编写翻译,假如现身某 .bat文本未找到,表达你的VS未有安装相应的信赖性(Linux下编写翻译不成事原因或然是不曾装python-dev),按小说开首给出的信任库增添修改(这时没有必要重新安装VS)。

编写翻译甘休后,在该文件夹下会现出 build 文件夹,进入该文件夹,现身如下多少个公文夹:

图片 3

进入 lib.xxx十一分文件夹,里面有个 .pyd 结尾的文本(Linux下为 .so 结尾),那便是我们编写翻译好的python模块了,如下:

图片 4

当然,你也得以改名字为 calc.pyd 相比较难堪,然而那不影响调用。

3. Python的参数字传送递以致C的再次回到值相关难题

那有个别自己直接甩出文件就行,编写翻译及调用进度与地点同样。

郑重声明:本文版权归美高梅163888所有,转载文章仅为传播更多信息之目的,如作者信息标记有误,请第一时间联系我们修改或删除,多谢。