Cgo:找不到使用带有 const char* 参数的回调的方法

Cgo: can't find way to use callbacks with const char* argument

我使用 Go 中的 C 库,使用 Cgo,除了回调之外一切都很好。库有回调 setter,它接受指向回调函数的指针。回调函数本身用 go 编写并使用 Cgo 语法导出。
问题:我可以使用 char * 参数创建和导出函数,但不能使用 const char *.

代码说明:
test.go:

package main

/*
typedef void (*cb_func)(const char *, int);
void callback(cb_func);
void myFunc(const char *, int);
*/
import "C"
import (
        "fmt"
        "unsafe"
)

//export myFunc
func myFunc(buf *C.char, ln C.int) {
        fmt.Printf("Got: %s\n", C.GoStringN(buf, ln))
}

func main() {
        C.callback((C.cb_func)(unsafe.Pointer(C.myFunc)))
}

test.c:

typedef void (*cb_func)(const char *, int);

void callback(cb_func cb) {
        cb("test", 4);
}

来自go build的输出:

In file included from $WORK/test/_obj/_cgo_export.c:2:0:
./test.go:54:13: error: conflicting types for 'myFunc'
./test.go:7:6: note: previous declaration of 'myFunc' was here
 void myFunc(const char *, int);
      ^
/tmp/go-build994908053/test/_obj/_cgo_export.c:9:6: error: conflicting types for 'myFunc'
 void myFunc(char* p0, int p1)
      ^
In file included from $WORK/test/_obj/_cgo_export.c:2:0:
./test.go:7:6: note: previous declaration of 'myFunc' was here
 void myFunc(const char *, int);
      ^

没有 const 限定符代码编译并按预期工作。

什么可以代替 *C.char 来获取 C 中的常量字符串?

由于 Go 没有 const 指针修饰符,因此无法从 Go 代码内部转换此行为。 cgo 将始终生成没有 const 修饰符的 headers。这也是您的代码无法正确构建的原因:cgo 仅根据它知道的内容创建 myFuncbuf 应该是 char*,而不是 const char* .

处理此问题的最佳方法是在 C 端使用包装器将该参数转换为 const char*。在您的情况下,将 myFunc 的定义更改为 void myFunc(char*, int) 就足够了。将函数传递给 cb_func 将起作用,因为将 myFunc 转换为 (*cb_func)(const char*,int) 只会添加类型信息,但不会更改内存布局。

我遇到了同样的问题。仍然需要的人的一种解决方案是:

typedef const char cchar_t
void myNewFunction(cchar_t* data);

然后在 go 中使用它作为 data* C.char_t