- r - 以节省内存的方式增长 data.frame
- ruby-on-rails - ruby/ruby on rails 内存泄漏检测
- android - 无法解析导入android.support.v7.app
- UNIX 域套接字与共享内存(映射文件)
我正在尝试使用 cgo 在 Go 中编写一个 MySQL UDF,其中我有一个基本的功能,但有一些我无法弄清楚的点点滴滴,因为我不知道一些 C 变量是什么是在围棋方面。
这是我用 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: invalid operation: (*args.arg_type)[0] (type uint32 does not support indexing)
我不太确定那是什么意思。这不应该是某种 slice ,而不是 uint32
吗?
这是 super 有用的地方,有一些方法可以将 args
结构转储到某个地方(甚至可能在 Go 语法中作为一个 super 加号),这样我就可以告诉我我在做什么与.
好吧,我使用 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>
})
最佳答案
好吧,@JimB 给了我很大的帮助,尽管我显然不太擅长 Go(尤其是 CGO),但我有一个可以工作的 UDF 版本,它简单直接(并且fast) 函数,从 URL 字符串中提取单个参数并正确解码它(例如 %20 作为空格返回,基本上是您期望它的工作方式)。
这对于纯 C UDF 来说似乎非常棘手,因为我并不真正了解 C(以及我了解其他语言),而且 URL 解析和 URL 参数解码以及 native MySQL 函数可能会出错很多是慢(而且也没有真正好的、干净的方法来进行解码),所以 Go 似乎是解决这类问题的完美候选者,因为它具有强大的性能、易于编写,以及各种易于使用的内置插件和第三方库。
完整的 UDF 及其安装/使用说明在这里 https://github.com/StirlingMarketingGroup/mysql-get-url-param/blob/master/main.go
第一个问题是调试输出。我通过 Fprintf
ing 到 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_RESULT
关于mysql - 如何在使用 cgo 构建时调试/转储 Go 变量?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/51446087/
我试图使用 cgo 为 x264 库编写一个小包装器,但遇到了嵌套结构的问题。该库使用了许多复杂的结构,其中一些字段本身就是匿名结构。 当尝试使用 cgo 访问这些结构时,我遇到了编译错误,因为 go
我正在使用一个 C 库,与下面不同,我无法控制它。我需要将指向也包含指针的数组的指针传递给 C 函数。 package main /* #include typedef int* pInt; voi
根据 CGO 的文档( https://pkg.go.dev/cmd/cgo ),在实现中有一个已知的错误: Note: the current implementation has a bug. W
我正在尝试使用 CGO 将现有的 Go 包构建到 C 共享库和 header 中. 我用 -buildmode c-shared 构建了这个包如记录。 -buildmode=c-shared
C部分: struct Person {...} struct Person * get_team(int * n) 转到部分: n := C.int(0) var team *C.struct_Pe
我在ubuntu 18上使用go版本go版本“go1.14.4 linux / amd64”。该代码不是我编写的,但我认为它可以与比当前版本早几年的go版本一起使用。当我尝试构建时,出现此错误。有什么
我编写了一个 Go 程序来模拟按键操作。为此,我必须使用 cgo 和不同的 C 代码片段,具体取决于正在编译 Go 代码的操作系统。我编写的代码如下所示: package keyboard /* #i
我已经在 Linux 中围绕信号量函数编写了一个包装器。这在过去的 Go 1.3 和 Go 1.4 上对我有用,但我需要使用这个包装器重建我的应用程序,它不再使用 Go 1.6.2 或 Go 1.7r
关注我的 earlier question ,现在我在尝试执行此 article 中的所有相同步骤时遇到此错误,因为我已经从 Go 1.6.1 升级到 Go 1.7.1(我无法返回到 Go 1.6.1
有时,C API 可能需要 NULL 指针。 这在 CGO 中可能吗? 例如,我想在 Go 语言程序中向 strcmp() 传递一个 null 参数: package strutil /* #incl
我正在使用 native linux C 二进制文件,它有一个相当昂贵的初始化调用,我想在应用程序启动时执行一次。这个调用应该在内部打开一堆文件句柄供以后使用。当我从 Go 调用这个昂贵的初始化 C
我正在使用这个库 https://github.com/billziss-gh/cgofuse ,还有一些接口(interface)需要实现,其中一个看起来像这样: func (self *Memfs
似乎我不能使用 Cgo 调用在另一个目录而不是当前 Go 包中声明的 C 函数。 所有文件的代码: // TestGoCallOC.go package main /* #include "test.
去项目结构 src test.go testc/collection/linkedlist.h testc/collection/linkedlist.c test.go package
我正在用 Go 编写一个库,我正在使用 CGo 将其编译为 native 库,但是当导出函数时,它们在函数声明和文档注释之间有这个烦人的空行。 例如,假设我的 go 文件中有这个 /** * Pri
为什么使用 Cgo 时我的 .go 文件中无法识别 c 函数? 我遵循了所有过程并尝试了 godoc 上的示例,它有效但这个不起作用,是什么原因? 文件夹结构 libsha.a sha.cpp 扫码
我正在用 cgo 包装一个 C 库,以供普通 Go 代码使用。 我的问题是我想将错误字符串向上传播到 Go API,但是有问题的 C 库通过线程本地存储提供错误字符串;有一个全局 get_error(
我有点乱,我看不出我是怎么摆脱它的……我得到了什么:* 我使用 cgo 构建的库 (dll)。* 使用此库 (dll) 的 C++ 应用程序。 我要做什么:我尝试释放在 C++ 应用程序的 cgo 库
我正在用 Go 编写一个库,我想导出到一个 c-shared-library。它工作得很好,但是我发现导出的 header 使用 p0、p1、p2、 有点烦人…… 用于参数名称,而不是 Go 中的原始
我目前正在开发一个程序,该程序使用带有 cgo 的 c 库调用长时间运行的函数。我无法使用 c 编辑库以允许超时。到目前为止,我唯一的解决方案是让僵尸 goroutine 继续运行 func Time
我是一名优秀的程序员,十分优秀!