系统:Mac OS 10.12.6。 Python:来自Anoconda3的Python 3.5.2。 用Cython == 0.28
我正在尝试为静态c ++ lib编写包装器。这是头文件的一部分。
/* LexActivator.h */
#pragma once
#include <stdint.h>
#ifdef _WIN32
#include <windows.h>
#ifdef LEXACTIVATOR_EXPORTS
#ifdef LEXACTIVATOR__STATIC
#define LEXACTIVATOR_API extern "C"
#else
#define LEXACTIVATOR_API extern "C" __declspec(dllexport)
#endif
#else
#ifdef __cplusplus
#ifdef LEXACTIVATOR_STATIC
#define LEXACTIVATOR_API extern "C"
#else
#define LEXACTIVATOR_API extern "C" __declspec(dllimport)
#endif
#else
#ifdef LEXACTIVATOR_STATIC
#define LEXACTIVATOR_API
#else
#define LEXACTIVATOR_API __declspec(dllimport)
#endif
#endif
#endif
#if defined(USE_STDCALL_DLL) && !defined(LEXACTIVATOR_STATIC)
#define LA_CC __stdcall
#else
#define LA_CC __cdecl
#endif
typedef const wchar_t* CSTRTYPE;
typedef wchar_t* STRTYPE;
#else
#define LA_CC
typedef int32_t HRESULT;
#if __GNUC__ >= 4
#ifdef __cplusplus
#define LEXACTIVATOR_API extern "C" __attribute__((visibility("default")))
#else
#define LEXACTIVATOR_API __attribute__((visibility("default")))
#endif
#else
#ifdef __cplusplus
#define LEXACTIVATOR_API extern "C"
#else
#define LEXACTIVATOR_API
#endif
#endif
typedef const char* CSTRTYPE;
typedef char* STRTYPE;
#endif
#define LA_USER ((uint32_t)1)
#define LA_SYSTEM ((uint32_t)2)
#define LA_V_TRIAL ((uint32_t)1)
#define LA_UV_TRIAL ((uint32_t)2)
LEXACTIVATOR_API HRESULT LA_CC SetProductFile(CSTRTYPE filePath);
这是Cython的pxd
文件的一部分。
from libc.stdint cimport *
cdef extern from "LexActivator.h":
ctypedef int32_t HRESULT
ctypedef const char* CSTRTYPE
ctypedef char* STRTYPE
uint32_t LA_USER = 1
uint32_t LA_SYSTEM = 2
uint32_t LA_V_TRIAL = 1
uint32_t LA_UV_TRIAL = 2
HRESULT SetProductFile(CSTRTYPE filePath)
我为测试编写了一个简单的pyx
文件。
cimport LexActivator
def SetProductFile(filePath):
cdef bytes py_bytes = filePath.encode()
cdef const char* c_string = py_bytes
cdef int32_t status = LexActivator.SetProductFile(c_string)
print(status)
return status
设置文件
from distutils.core import setup
from distutils.extension import Extension
from Cython.Build import cythonize
ext_modules=[
Extension("LexActivator",
sources=["LexActivator.pyx"],
language='c++',
extra_objects=["libLexActivator.a"],
)
]
setup(
name = "LexActivator",
ext_modules = cythonize(ext_modules)
)
运行python setup.py build_ext --inplace
。
Error compiling Cython file:
------------------------------------------------------------
...
cimport LexActivator
def SetProductFile(filePath):
cdef bytes py_bytes = filePath.encode()
cdef const char* c_string = py_bytes
cdef int32_t status = LexActivator.SetProductFile(c_string)
^
------------------------------------------------------------
LexActivator.pyx:7:38: cimported module has no attribute 'SetProductFile'
PS:我已经成功地使用Xcode完成了这个代码。
答案 0 :(得分:0)
我无法重现您的确切错误,但我认为问题是您要覆盖SetProductFile
。
pxd
文件同名的 pyx
文件会自动被导入 - 相当于from LexActivator cimport *
。因此,您不需要cimport LexActivator
。出现此问题是因为def
函数SetProductFile
与您的C函数具有相同的名称(这会让Cython感到困惑 - 我会收到警告而不是错误)
我也会避免使用Cython和C文件中的相同名称。 Cython编译会生成LexActivator.c
文件,在某些情况下会生成LexActivator.h
文件,因此如果它们具有相同的名称,您的C文件很容易被覆盖。
总之,您有两种选择:
重命名您的Python def
函数(并删除cimport LexActivator
)
(正如@ead建议的那样)将您的pxd
重命名为其他内容,执行cimport SomethingElse
,然后将SetProductFile
赢得冲突名称。