为了账号安全,请及时绑定邮箱和手机立即绑定

从 Python 调用 Go

从 Python 调用 Go

Go
斯蒂芬大帝 2023-06-26 15:41:27
我尝试从 python 运行一个简单的 go 脚本,但出现了分段错误。这是我的代码:主程序package mainimport (    /*typedef struct foo{int a;int b;int c;int d;int e;int f;} foo;*/    "C")func main() {}//export Foofunc Foo(t []int) C.foo {    return C.foo{}}main.py# loading shared objectlib = cdll.LoadLibrary("main.so")# go typeclass GoSlice(Structure):    _fields_ = [("data", POINTER(c_void_p)), ("len", c_longlong), ("cap", c_longlong)]lib.Foo.argtypes = [GoSlice]lib.Foo.restype = c_void_pt = 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现在如果我删除e并f从main.go我得到None并且不再出现分段错误。为什么结构中的成员数量在这里很重要?[编辑] 两者都在同一位置运行,我运行一个命令clear && go build -o main.so -buildmode=c-shared main.go && python3 main.py
查看完整描述

1 回答

?
慕神8447489

TA贡献1780条经验 获得超1个赞

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


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 {

    foo := C.foo{}

    foo.a = 1 // setting some values to avoid seeing zeros

    foo.b = 2

    return foo

}

主要.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


查看完整回答
反对 回复 2023-06-26
  • 1 回答
  • 0 关注
  • 118 浏览
慕课专栏
更多

添加回答

举报

0/150
提交
取消
意见反馈 帮助中心 APP下载
官方微信