我正在尝试通过cgo使用C函数strfmon
。
有效的示例C代码是:
#include <stdio.h>
#include <monetary.h>
int main(void)
{
char str[100];
double money = 1234.56;
strfmon(str, 100, "%i", money);
printf("%s\n", string);
}
到目前为止,我编写的Go代码是:
package main
// #cgo CFLAGS: -g -Wall
// #include <stdlib.h>
// #include <monetary.h>
import "C"
import (
"fmt"
)
func main() {
str := [100]C.char{}
var money C.double = 1234.56
C.strfmon(str, 100, "%i", money)
fmt.Printf("%+v\n", str)
}
go run main.go
出现以下错误:
./main.go:14:2: unexpected type: ...
我相信...
指的是strfmon
中的可变参数,但我不确定如何从Go中解决。
答案 0 :(得分:2)
不支持调用可变C函数。可以通过使用C函数包装程序来规避这一点。
并且strfmon(3p)
确实是可变函数,如签名中的...
字符所示:
ssize_t strfmon(char *restrict s, size_t maxsize,
const char *restrict format, ...);
这样,您可以在C中创建一个包装函数,该函数具有固定数量的参数,并根据需要调用strfmon(...)
,例如:
package main
// #cgo CFLAGS: -g -Wall
//
// #include <locale.h>
// #include <monetary.h>
// #include <stdlib.h>
//
// size_t format_amount(char * s, size_t maxsize, char * format, double amount)
// {
// setlocale(LC_ALL, "en_US");
// return strfmon(s, maxsize, format, amount);
// }
//
import "C"
import "fmt"
import "unsafe"
const SIZE = 100
func main() {
str := C.CString(string(make([]byte, SIZE)))
money := C.double(1234.56)
format := C.CString("[%n]")
C.format_amount(str, SIZE-1, format, money) // Call our wrapper here.
fmt.Printf("OK: %s\n", C.GoString(str))
// OK: [$1,234.56]
C.free(unsafe.Pointer(str))
C.free(unsafe.Pointer(format))
}