我试图在Go和cgo一起编写一个MySQL UDF,其中我有一个基本的功能,但是有一些小的零碎我无法弄清楚,因为我不知道C变量中有哪些是Go的。
这是我用C编写的一个示例,它强制一个MySQL参数的类型为int。
my_bool unhex_sha3_init(UDF_INIT *initid, UDF_ARGS *args, char *message) {
if (args->arg_count != 2) {
strcpy(message, "`unhex_sha3`() requires 2 parameters: the message part, and the bits");
return 1;
}
args->arg_type[1] = INT_RESULT;
initid->maybe_null = 1; //can return null
return 0;
}这很好,但是我试着用Go中的另一个函数做同样的/类似的事情。
//export get_url_param_init
func get_url_param_init(initid *C.UDF_INIT, args *C.UDF_ARGS, message *C.char) C.my_bool {
if args.arg_count != 2 {
message = C.CString("`get_url_param` require 2 parameters: the URL string and the param name")
return 1
}
(*args.arg_type)[0] = C.STRING_RESULT
(*args.arg_type)[1] = C.STRING_RESULT
initid.maybe_null = 1
return 0
}使用此生成错误
./main.go:24:无效操作:(*args.arg_type)0
我不太清楚那是什么意思。这难道不是某种类型的片段,而不是uint32
这将是非常有用的地方,有一些方法可以在某种程度上转储args结构(甚至在Go语法中也是超级加号),这样我就可以知道我在做什么了。
我使用spew将变量内容转储到init函数中的tmp文件中(注释掉使其不编译的行),并得到以下结果
(string) (len=3) "%#v"
(*main._Ctype_struct_st_udf_args)(0x7ff318006af8)({
arg_count: (main._Ctype_uint) 2,
_: ([4]uint8) (len=4 cap=4) {
00000000 00 00 00 00 |....|
},
arg_type: (*uint32)(0x7ff318006d18)(0),
args: (**main._Ctype_char)(0x7ff318006d20->0x7ff3180251b0)(0),
lengths: (*main._Ctype_ulong)(0x7ff318006d30)(0),
maybe_null: (*main._Ctype_char)(0x7ff318006d40)(0),
attributes: (**main._Ctype_char)(0x7ff318006d58->0x7ff318006b88)(39),
attribute_lengths: (*main._Ctype_ulong)(0x7ff318006d68)(2),
extension: (unsafe.Pointer) <nil>
})发布于 2018-07-20 19:45:07
好吧,@JimB对我有很大的帮助,尽管我对Go (尤其是CGO)显然不太熟悉,但我有一个UDF的工作版本,它是一个简单、直接(且快速)的函数,它从URL字符串中提取单个参数,并正确地解码它和不正确的内容(例如,%20作为空格返回,基本上是您期望它工作的方式)。
对于纯C UDF来说,这看起来非常棘手,因为我并不真正了解C(以及其他语言),而且URL解析和URL参数解码有很多问题,本地的MySQL函数也很慢(而且也没有一种很好、干净的解码方法),因此Go似乎是解决这类问题的最佳人选,因为它具有很强的性能、易于编写,以及各种易于使用的内置内置和第三方库。
完整的UDF及其安装/使用说明如下:https://github.com/StirlingMarketingGroup/mysql-get-url-param/blob/master/main.go
第一个问题是调试输出。我是通过Fprintf对一个tmp文件而不是标准输出来实现这一点的,这样我就可以检查该文件以查看变量转储。
t, err := ioutil.TempFile(os.TempDir(), "get-url-param")
fmt.Fprintf(t, "%#v\n", args.arg_type)然后,在得到输出后(我希望args.arg_type是一个数组,就像它在C中那样,而是一个数字),我需要将这个数字所引用的数据(指向C数组开始的指针)转换成一个Go数组,以便设置它的值。
argsTypes := *(*[2]uint32)(unsafe.Pointer(args.arg_type))
argsTypes[0] = C.STRING_RESULT
argsTypes[1] = C.STRING_RESULThttps://stackoverflow.com/questions/51446087
复制相似问题