Cython无法从cimported模块中找到声明的函数

时间:2018-03-17 21:58:04

标签: cython cythonize

系统: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完成了这个代码。

1 个答案:

答案 0 :(得分:0)

我无法重现您的确切错误,但我认为问题是您要覆盖SetProductFile

pxd文件同名的

pyx文件会自动被导入 - 相当于from LexActivator cimport *。因此,您不需要cimport LexActivator。出现此问题是因为def函数SetProductFile与您的C函数具有相同的名称(这会让Cython感到困惑 - 我会收到警告而不是错误)

我也会避免使用Cython和C文件中的相同名称。 Cython编译会生成LexActivator.c文件,在某些情况下会生成LexActivator.h文件,因此如果它们具有相同的名称,您的C文件很容易被覆盖。

总之,您有两种选择:

  1. 重命名您的Python def函数(并删除cimport LexActivator

  2. (正如@ead建议的那样)将您的pxd重命名为其他内容,执行cimport SomethingElse,然后将SetProductFile赢得冲突名称。

    < / LI>