
Go 1.5 引入了构建可从 C 调用的共享库的支持,这使得使用 Go 编写 Ruby 扩展变得更加容易。通过 Ruby 的 FFI (Foreign Function Interface) 库,我们可以直接调用 Go 编译的共享库中的函数,而无需编写额外的 C 胶水代码。这为利用 Go 语言的并发特性和性能优势来增强 Ruby 应用提供了可能。
首先,我们创建一个 Go 源文件 goFuncs.go,其中包含我们想要暴露给 Ruby 调用的函数。
package main
import "C"
//export GoAdd
func GoAdd(a, b C.int) C.int {
return a + b
}
func main() {} // Required but ignored关键点:
- import "C": 必须导入 "C" 包,以便能够使用 C 的数据类型。
- //export GoAdd: //export 注释是必需的,它告诉 go build 命令将 GoAdd 函数导出为共享库中的一个符号。export 后的符号名称是该函数在共享库中的名称,Ruby 将使用该名称来调用它。
- func main() {}: 即使我们不使用 main 函数,也必须定义它。这是 Go 编译器的要求。
接下来,我们需要使用 go build 命令将 Go 代码编译成共享库。
立即学习“go语言免费学习笔记(深入)”;
go build -buildmode=c-shared -o goFuncs.so goFuncs.go
命令解释:
- go build: Go 编译器命令。
- -buildmode=c-shared: 指定构建模式为 c-shared,这意味着我们将构建一个可以被 C 代码调用的共享库。
- -o goFuncs.so: 指定输出文件名为 goFuncs.so。这个文件将是我们的共享库。
- goFuncs.go: 指定要编译的 Go 源文件。
现在,我们创建一个 Ruby 文件 goFromRuby.rb,用于调用 Go 共享库中的函数。
在商业版本基础上修改发布的,无功能限制,无使用时间限制,功能全面扩展,版面美观大方,俱备完整的购物网站功能、购物方便,操作简单,在线支付功能,采用网银支付完美在线支付接口,对一些文件重新编写,减少了图片的使用,网站整体配色更加明朗v2.01修正2005个人免费版出现的问题
require 'ffi' module GoFuncs extend FFI::Library ffi_lib './goFuncs.so' attach_function :GoAdd, [:int, :int], :int end puts GoFuncs.GoAdd(41, 1)
代码解释:
- require 'ffi': 导入 FFI 库。
- module GoFuncs: 定义一个 Ruby 模块,用于封装 Go 函数。
- extend FFI::Library: 将 FFI::Library 模块的功能扩展到 GoFuncs 模块。
- ffi_lib './goFuncs.so': 指定要加载的共享库文件。
- attach_function :GoAdd, [:int, :int], :int: 将 Go 函数 GoAdd 绑定到 Ruby。第一个参数是 Ruby 中用于调用 Go 函数的名称(GoAdd),第二个参数是 Go 函数的参数类型列表(两个 int),第三个参数是 Go 函数的返回值类型(int)。
- puts GoFuncs.GoAdd(41, 1): 调用 Go 函数 GoAdd,并将结果打印到控制台。
最后,运行 Ruby 脚本:
ruby goFromRuby.rb
输出结果应该是:
42
注意事项:
- 确保安装了 FFI 库。可以使用 gem install ffi 命令安装。
- 共享库文件 (goFuncs.so) 必须位于 Ruby 脚本可以访问的位置。
- 类型匹配非常重要。确保 Ruby 中声明的参数类型和返回值类型与 Go 函数中的类型匹配。
- 在更复杂的场景中,可能需要处理 Go 和 Ruby 之间的数据类型转换。
总结:
通过 Go 提供的 c-shared 构建模式和 Ruby 的 FFI 库,我们可以轻松地使用 Go 编写 Ruby 扩展。这为利用 Go 的高性能和并发特性来提升 Ruby 应用的性能提供了便捷的途径。虽然本文展示了一个简单的加法函数的例子,但这种方法可以扩展到更复杂的场景,例如处理网络请求、执行计算密集型任务等。在实际应用中,需要仔细考虑数据类型转换和错误处理,以确保 Go 和 Ruby 之间的平滑集成。









