我一直在用C语言编写Python模块。我从一个计算两点规范的简单例子开始。代码看起来像这样,
_norm.c
#include <Python.h>
#include "norm.h"
static char module_docstring[] =
"This module provides an interface for computing the norm using C.";
static char norm_docstring[] =
"Calculate the norm between two points";
static char norm2_docstring[] =
"Calculate the square norm between two points. For efficiency.";
static PyObject *norm_norm(PyObject *self, PyObject *args);
static PyObject *norm_norm2(PyObject *self, PyObject *args);
static PyMethodDef module_methods[] = {
{"norm", norm_norm, METH_VARARGS, norm_docstring},
{"norm2", norm_norm2, METH_VARARGS, norm2_docstring},
{NULL, NULL, 0, NULL}
};
PyMODINIT_FUNC init_norm(void) {
PyObject *m = Py_InitModule3("_norm", module_methods, module_docstring);
if (m == NULL)
return;
}
static PyObject *norm_norm(PyObject *self, PyObject *args) {
double x1, y1, x2, y2;
/* Parse the input tuple */
if (!PyArg_ParseTuple(args, "ddOOO", &x1, &y1, &x2, &y2))
return NULL;
/* Call the external C function to compute the norm. */
double value = norm(x1, y1, x2, y2);
if (value < 0.0) {
PyErr_SetString(PyExc_RuntimeError,
"Norm returned an impossible value.");
}
PyObject *ret = Py_BuildValue("d", value);
return ret;
}
norm.c
#include <math.h>
long double norm2(long double x1, long double y1, long double x2, long double y2) {
long double xd = x2 - x1;
long double yd = y2 - y1;
return xd * xd + yd * yd;
}
long double norm(long double x1, long double y1, long double x2, long double y2) {
return sqrt(norm2(x1, y1, x2, y2));
}
setup.py
from distutils.core import setup, Extension
setup(
ext_modules=[Extension("_norm", ["_norm.c", "norm.c"])]
)
我像这样构建包,
python setup.py build_ext --inplace
它编译没有任何问题。但是,当我尝试使用它时,我得到一个关于参数数量的错误。
>>> import _norm
>>> _norm.norm(1,2,5,6)
Traceback (most recent call last):
File "<stdin>", line 1, in <module>
TypeError: function takes exactly 5 arguments (4 given)
我想这可能与PyObject * norm_norm()的声明有关,因为它传递了* self,但是我不确定这是否会影响我传递给Python上的模块的参数事情的一面。我很感激任何帮助/建议。
答案 0 :(得分:3)
PyArg_ParseTuple
格式字符串中的问题。您正在尝试提取四个双参数,但格式字符串用于两个双参数和三个任意Python对象(&#34; ddOOO&#34;)。
正确的格式字符串应为&#34; dddd&#34;为了你想做什么。 改变这个
if(!PyArg_ParseTuple(args,&#34; ddOOO&#34;,&amp; x1,&amp; y1,&amp; x2,&amp; y2))
带
if(!PyArg_ParseTuple(args,&#34; dddd&#34;,&amp; x1,&amp; y1,&amp; x2,&amp; y2))