登录
首页 >  Golang >  Go问答

python调用cgo库——32位和64位问题

来源:stackoverflow

时间:2024-04-05 18:00:33 248浏览 收藏

欢迎各位小伙伴来到golang学习网,相聚于此都是缘哈哈哈!今天我给大家带来《python调用cgo库——32位和64位问题》,这篇文章主要讲到等等知识,如果你对Golang相关的知识非常感兴趣或者正在自学,都可以关注我,我会持续更新相关文章!当然,有什么建议也欢迎在评论留言提出!一起学习!

问题内容

我构建了一个cgo库并使用python(带有ctypes包)来调用它。代码被编译成32位和64位版本,这些库分别由32位和64位python程序调用。我发现显然参数没有正确传递。我认为这可能与数组在 python 程序和库之间的定义和传递方式有关。

例如,go库中的函数“callnames.so”定义为

func initialize(namelist []*c.char, grp *c.char)

调用该函数的python代码部分是 goslicechar 类(结构): 字段 = [("data", pointer(c_char_p)), ("len", c_longlong), ("cap", c_longlong)]

numnames = 3
n1 = c_char_p(b"peter")
n2 = c_char_p(b"tom")
n3 = c_char_p(b"nancy")
group = c_char_p(b"group1")

names = goslicechar((c_char_p * numcomponents)(n1, n2, n3), numnames, numnames ) 


lib = cdll.loadlibrary("./callnames.so")
lib.initialize(names,  group)

这些代码在64位环境下工作得很好,即python-64 + 64位cgo库。然而,当我切换到32位时,问题就出现了。我通过将 python 中 goslicechar 的定义更改为

进行了快速而肮脏的修复
class GoSliceChar(Structure):
       _fields_ = [("data", POINTER(c_char_p)), ("len", c_long), ("cap", c_long)]

但我不太明白为什么问题解决了,是否是一个可靠的解决方案。请帮忙。谢谢。


解决方案


我对我的问题有了一些新发现。看起来编译 cgo 库后,会自动生成一个 .h 文件。在这个文件中,所有的数组都被定义为goslice

typedef struct { void *data; goint len; goint cap; } goslice;

对于 32 位库,goint 定义为

typedef goint32 goint;

而对于 64 位库,goint 定义为

typedef goint32 goint;

在我看来,必须确保 python 代码在调用这些库时能够适应这一点

class GoSliceChar(Structure):
    if is_64bit():
        _fields_ = [("data", POINTER(c_char_p)), ("len", c_longlong), ("cap", c_longlong)]  
    else:
        _fields_ = [("data", POINTER(c_char_p)), ("len", c_long), ("cap", c_long)]

以上就是《python调用cgo库——32位和64位问题》的详细内容,更多关于的资料请关注golang学习网公众号!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>