python调用golang并回调
最近折腾python交互,也真够呛的,一连玩了好几天,被虐的不要不要的。天天各种百度,Google之间。
好吧,废话少说,转入我们的正题。其实,py调用go一般的函数,只是第一道坎,正主其实是py调用go,并且go还回调py!!!
网上其实这些问题很少,而且有且只有一篇关于go回调py的。
就是如下一位大兄弟写的:https://www.golangtc.com/t/59f858c04ce40d3bf47f5fbc
虽说他是说解决了,问题是下面的解决方案写的真心有问题啊。。。其实,py调用go,他们是通过c来进行桥接(应该是这么说吧),py<——>c<——>go,就是说,py一直认为自己是调用c,go也是如此,并不知其实他们是在互相操作。。。
那么,好办了,py调用go并且回调,在py侧,只要按照py调用c,并且回调就可以了。go侧则go调用c,并且回调c,就可以了。
其实py侧很简单,随便百度一下,应该是正确的。上面那大兄弟写的方法就可以了。问题是go侧,真心坑。。。
当py传入自己的回调,其实是被c包装了一下,然后,go这边接收的其实就是一个c的函数指针!在go里面,c的函数指针,其实就是一个unsafe.Pointer,一个unsafe.Pointer,一个unsafe.Pointer,重要的事情说三次!!!但go获得这东西,它只知道是一个地址啊,不知道是一个什么东西。。。好吧,只能把它重新转回c的函数指针,但这个过程必须要靠一个c函数做过渡!!!!
然而,这样就出现一个坑了!!!那个c函数定义,居然不能跟导出的go函数写在同一个go文件里面!!!否则,会一直报重复定义的错误,呵呵。于是乎,只能这么弄,分三个文件:一个.h文件,两个.go文件。
clib.h
#ifndef CLIB_H
#define CLIB_H
typedef void (*callback)(int);
#endif
这是定义回调结构的。
clibh.go
package main
import "C"
这是定义go调用c函数的,而且这个必须要有,用来间接调用c回调(py回调)的。
main.go
package main
import "C"
import (
"unsafe"
)
//export TestCB
func TestCB(a, b int, cb unsafe.Pointer){
c:=a+b
C.TestCCB(C.int(c), (*[0]byte)(cb))
}
func main() {
}
然后这个,是导出我们的正主:TestCB。其中的参数,cb就是针对c(py)回调的,在函数体里面,其实用TestCCB(中间c函数)来调用这个回调,注意:上方extern void TestCCB(int c, callback cb);只能这么弄了,不能直接在这个.go文件写它的定义。我就是为此折腾了好些天的,直接在里面定义c中间函数,就直接报重复定义了。
然后,编译命令要注意了:
go build -ldflags=-s -buildmode=c-shared -o foo.so clibh.go main.go
两个.go文件,必须要写出来!就是上面那大兄弟说的连个go,当时我看着他说的这个名词,愣了半天,不知他说啥。。。其实就是把该编译的go都写吧。。。不知为啥go编译的时候,不会主动把同一个包的代码都编译在一起。。。随便,能用就行。。。
接下来,在py侧,就很简单了。。。
import ctypes
lib = ctypes.CDLL('./foo.so')
CGOFunc = ctypes.CFUNCTYPE(None, ctypes.c_int32)
def GoCB(c):
print("c =", c)
cb = CGOFunc(GoCB)
lib.TestCB(5, 6, cb)
这个网上都有说,其实就是py调用c接口,然后c又回调py函数的做法。。。以上。
参考了几个网址:
Https://www.golangtc.com/t/59f858c04ce40d3bf47f5fbc
https://GitHub.com/golang/go/wiki/cgo#function-pointer-callbacks
https://xiaowing.github.io/post/howto_call_a_go_func_via_funcpoint_from_cside/
https://studygolang.com/articles/2629
相关文章