所以我正在研究Ox和Python之间的接口。问题是:Ox定义了一个函数c_pow
,Cython'mangles'到_Py_c_pow
。不幸的是,这个函数是allready defined。在这种情况下,有没有办法防止命名冲突?我可以指定自己的前缀吗?
#"jdmath.h"
...
void JDCALL c_pow(double xr, double xi, double a, double *yr, double *yi);
...
#"oxcy.pxd"
cdef extern from r'dev\jdmath.h':
pass
编译器输出:
PS D:\Git\OxPy\oxcy> python setup.py build_ext --inplace
running build_ext
cythoning oxcy.pyx to oxcy.c
building 'oxcy' extension
~\AppData\Local\Programs\Common\Microsoft\Visual C++ for Python\9.0\VC\Bin\amd64\cl.exe /c /nologo /Ox /MD /W3 /GS- /DNDEBUG "-IC:\Program Files (x86)\OxMetrics7\ox\bin" "-IC:\Program Files (x86)\OxMetrics7\ox\dev\lib64" "-IC:\Program Files (x86)\OxMetrics7\ox\dev" -IC:\Anaconda2\include -IC:\Anaconda2\PC /Tcoxcy.c /Fobuild\temp.win-amd64-2.7\Release\oxcy.obj
oxcy.c
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4031: second formal parameter list longer than the first list
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4028: formal parameter 1 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(111) : warning C4273: '_Py_c_abs' : inconsistent dll linkage
c:\anaconda2\include\complexobject.h(30) : see previous definition of '_Py_c_abs'
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4031: second formal parameter list longer than the first list
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4028: formal parameter 1 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(117) : warning C4028: formal parameter 2 different from declaration
d:\git\oxpy\oxcy\dev\jdmath.h(117) : error C2371: '_Py_c_pow' : redefinition; different basic types
c:\anaconda2\include\complexobject.h(29) : see declaration of '_Py_c_pow'
error: command '~\\AppData\\Local\\Programs\\Common\\Microsoft\\Visual C++ for Python\\9.0\\VC\\Bin\\amd64\\cl.exe' failed with exit status 2
事实证明,Anaconda定义了一个宏:#define c_pow _Py_c_pow
。因此,尽管Cython不应该归咎于修改,但编译仍然会因Python C头中的宏和来自Ox的头部引起的名称冲突而失败。我该如何规避这个问题呢?
答案 0 :(得分:1)
正如您所发现的,这是由导致重命名的macros defined by Python 2引起的。它们未在Python 3中定义,因此一个选项是升级。
假设您不想升级(这可能是合理的......),那就更难了。最简单的尝试就是添加
#undef c_pow
#undef c_abs
到jdmath.h的顶部。 (或者,如果你不能修改jdmath.h然后创建一个新的头,它取消定义宏并包含jdmath.h然后使用来自Cython的新头)。
希望这样可以正常工作。然而,如果它试图使用Python定义的c_pow
和c_abs
本身,那么这可能会破坏Cython中的某些东西(你会得到明显的编译错误)。在这种情况下,您需要创建一个使用新名称创建函数的标题:
// undefine so we can access c_pow and c_abs
#undef c_pow
#undef c_abs
#include "dev/jdmath.h"
inline void not_c_pow(/* fill in args yourself*/) { c_pow(/*args*/); }
inline void not_c_abs(/* fill in args yourself*/) { c_abs(/*args*/); }
// redefine to original state
#define c_pow _Py_c_pow
#define c_abs _Py_c_abs
在Cython中你可以使用not_c_pow
和not_c_abs
来代替它,它应该有用。