python 从扩展模块中定义和导出C的API

C语言定义函数库API

  • C语言定义函数头源代码,sample.h
/*
 * sample.h
 *
 *  Created on: 2022年12月29日
 *      Author: xzlAwin
 */
extern int gcd(int x, int y);
extern int in_mandel(double x0, double y0, int n);
extern int divide(int a, int b, int *remainder);
extern double avg(double *a, int n);

typedef struct Point {
	double x,y;
} Point;

extern double distance(Point *p1, Point *p2);
  • C语言实现定义函数源代码,sample.c
/*
 * sample.c
 *
 *  Created on: 2022年12月29日
 *      Author: xzlAwin
 */

#include<math.h>

/* Compute the greatest common divisor */
int gcd(int x, int y) {
	int g = y;
	while(x > 0) {
		g = x;
		x = y % x;
		y = g;
	}
	return g;
}

/* Test if (x0,y0) is in the Mandelbrot set or not */
int in_mandel(double x0, double y0, int n) {
	double x=0,y=0,xtemp;
	while (n > 0) {
		xtemp = x*x - y*y +x0;
		y = 2*x*y + y0;
		x = xtemp;
		n -= 1;
		if(x*x + y*y > 4) return 0;
	}
	return 1;
}

/* Divide two numbers */
int divide(int a, int b, int *remainder) {
	int quot = a / b;
	*remainder = a % b;
	return quot;
}

/* Average values in an array */
double avg(double *a, int n) {
	int i;
	double total = 0.0;
	for (i = 0; i < n; i++) {
		total += a[i];
	}
	return total / n;
}

/* A C data structure */
typedef struct Point {
	double x,y;
} Point;

/* Function involving a C data structure */
double distance(Point *p1, Point *p2) {
	return hypot(p1->x - p2->x, p1->y - p2->y);
}
  • C语言暴露函数库API源代码,pysample.c
/*
 * pysample.h
 *
 *  Created on: 2022年12月30日
 *      Author: xzlAwin
 */

#include "Python.h"
#include "sample.h"
#ifdef __cplusplus
extern "C" {
#endif

/* Public API Table */
typedef struct {
	Point *(*aspoint)(PyObject *);
	PyObject *(*frompoint)(Point *, int);
} _PointAPIMethods;

#ifndef PYSAMPLE_MODULE
/* Method table in external module */
static _PointAPIMethods *_point_api = 0;

/* Import the API table from sample */
static int import_sample(void) {
	_point_api = (_PointAPIMethods *) PyCapsule_Import("sample._point_api",0);
	return (_point_api != NULL) ? 1 : 0;
}

/* Macros to implement the programming interface */
#define PyPoint_AsPoint(obj) (_point_api->aspoint)(obj)
#define PyPoint_FromPoint(obj) (_point_api->frompoint)(obj)
#endif

#ifdef __cplusplus
}
#endif
  • python编译配置,setup.py
from distutils.core import setup, Extension

setup(name="sample",
    ext_modules=[
        Extension(
            "sample",
            ["sample.c", "pysample.c"],
            include_dirs=[".."]
        )
    ]
)
  • python命令行,编译函数库,编译成功生成 编译成功生成 sample.cp37-win_amd64.pyd
python3 setup.py build_ext --inplace

扩展模块中的定义

  • 扩展模块头源代码,pysample.h
/*
 * pysample.h
 *
 *  Created on: 2022年12月30日
 *      Author: xzlAwin
 */

#include "Python.h"
#include "sample.h"
#ifdef __cplusplus
extern "C" {
#endif

/* Public API Table */
typedef struct {
	Point *(*aspoint)(PyObject *);
	PyObject *(*frompoint)(Point *, int);
} _PointAPIMethods;

#ifndef PYSAMPLE_MODULE
/* Method table in external module */
static _PointAPIMethods *_point_api = 0;

/* Import the API table from sample */
static int import_sample(void) {
	_point_api = (_PointAPIMethods *) PyCapsule_Import("sample._point_api",0);
	return (_point_api != NULL) ? 1 : 0;
}

/* Macros to implement the programming interface */
#define PyPoint_AsPoint(obj) (_point_api->aspoint)(obj)
#define PyPoint_FromPoint(obj) (_point_api->frompoint)(obj)
#endif

#ifdef __cplusplus
}
#endif
  • 扩展模块源代码,ptexample.c
/*
 * ptexample.c
 *
 *  Created on: 2022年12月30日
 *      Author: xzlAwin
 */

/* Include the header associated with the other module */
#include "pysample.h"

/* An extension function that uses the exported API */
static PyObject *print_point(PyObject *self, PyObject *args) {
	PyObject *obj;
	Point *p;
	if (!PyArg_ParseTuple(args,"O", &obj)) {
		return NULL;
	}

	/* Note: This is defined in a different module */
	p = PyPoint_AsPoint(obj);
	if (!p) {
		return NULL;
	}
	printf("%f %f\n", p->x, p->y);
	return Py_BuildValue("");
}

static PyMethodDef PtExampleMethods[] = {
	{"print_point", print_point, METH_VARARGS, "output a point"},
	{ NULL, NULL, 0, NULL}
};

static struct PyModuleDef ptexamplemodule = {
	PyModuleDef_HEAD_INIT,
	/* name of module */
	"ptexample",
	/* Doc string (may be NULL) */
	"A module that imports an API",
	/* Size of per-interpreter state or -1 */
	-1,
	/* Method table */
	PtExampleMethods
};

/* Module initialization function */
PyMODINIT_FUNC
PyInit_ptexample(void) {
	PyObject *m;

	m = PyModule_Create(&ptexamplemodule);
	if (m == NULL)
		return NULL;

	/* Import sample, loading its API functions */
	if (!import_sample()) {
		return NULL;
	}
	return m;
}
  • python编译配置,ptsetup.py
from distutils.core import setup, Extension

setup(name="sample",
    ext_modules=[
        Extension(
            "ptexample",
            ["ptexample.c"],
            include_dirs=["."]
        )
    ]
)
  • python命令行,编译函数库,编译成功生成 编译成功生成 ptexample.cp37-win_amd64.pyd
python3 ptsetup.py build_ext --inplace

测试

  • python调用库和扩展模块,main.py
import sample
import ptexample

p1 = sample.Point(2, 3)
print(p1)
ptexample.print_point(p1)