【问题标题】:Calling Go from Python从 Python 调用 Go
【发布时间】:2019-10-28 09:19:30
【问题描述】:

我尝试在 python 的 go 脚本中运行一个简单的脚本,但遇到了分段错误。这是我的代码:

main.go

package main

import (
    /*
typedef struct foo{
int a;
int b;
int c;
int d;
int e;
int f;
} foo;
*/
    "C"
)

func main() {}

//export Foo
func Foo(t []int) C.foo {
    return C.foo{}
}

main.py

# loading shared object
lib = cdll.LoadLibrary("main.so")

# go type
class GoSlice(Structure):
    _fields_ = [("data", POINTER(c_void_p)), ("len", c_longlong), ("cap", c_longlong)]

lib.Foo.argtypes = [GoSlice]
lib.Foo.restype = c_void_p

t = GoSlice((c_void_p * 5)(1, 2, 3, 4, 5), 5, 5)
f = lib.Foo(t)
print(f)

使用该代码,我得到了

140362617287784
[1]    23067 segmentation fault  python3 main.py

现在如果我从 main.go 中删除 ef 我得到了

None

不再有分段错误。

为什么结构中的成员数量在这里很重要?

[编辑] 两者都在同一个地方运行,我运行一个命令clear && go build -o main.so -buildmode=c-shared main.go && python3 main.py

【问题讨论】:

  • 你确定你的 Python 安装和 Go 生成的库都是 32 位还是 64 位?
  • @georgeok 你好,两者都在我的电脑上运行在同一个地方,只有一个命令clear && go build -o main.so -buildmode=c-shared main.go && python3 main.py

标签: python python-3.x go cgo


【解决方案1】:

您的 GO/C 代码是正确的。问题出在python脚本中。 lib.Foo.restype = c_void_p 调用需要一个 void 指针,但库返回一个 C 结构。您需要在 python 中定义一个 ctypes 结构的返回类型,然后它将按您的预期工作。

ma​​in.go

package main

import (
    /*
    typedef struct foo{
    int a;
    int b;
    int c;
    int d;
    int e;
    int f;
    } foo;
    */
    "C"
)

func main() {
}

//export Foo
func Foo(t []int) C.foo {
    foo := C.foo{}
    foo.a = 1 // setting some values to avoid seeing zeros
    foo.b = 2
    return foo
}

ma​​in.py

from ctypes import *

# loading shared object
lib = cdll.LoadLibrary("main.so")


# go type
class GoSlice(Structure):
    _fields_ = [("data", POINTER(c_void_p)), ("len", c_longlong), ("cap", c_longlong)]


class Foo(Structure):
    _fields_ = [('a', c_int),
                ('b', c_int),
                ('c', c_int),
                ('d', c_int),
                ('e', c_int),
                ('f', c_int)]


lib.Foo.argtypes = [GoSlice]
lib.Foo.restype = Foo

t = GoSlice((c_void_p * 5)(1, 2, 3, 4, 5), 5, 5)
f = lib.Foo(t)
print(f)
print(f.a)
print(f.b)

然后运行go build -o main.so -buildmode=c-shared main.go && python main.py,它会打印:

go build -o main.so -buildmode=c-shared main.go && python3 main.py 
<__main__.Foo object at 0x102608830>
1
2

【讨论】:

  • 有效!万分感谢。实际上,我在 stackoverflow 上阅读了另一个主题(我不再找到它),它解释了(如果我很好理解的话)c_void_p 足以作为处理结构的返回类型。而且由于它适用于我程序的另一个结构,我认为这是正确的方法。感谢您澄清这一点。
猜你喜欢
  • 1970-01-01
  • 1970-01-01
  • 2011-09-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 1970-01-01
  • 2013-10-24
  • 2019-09-18
相关资源
最近更新 更多