У меня есть функция в Go:

func login(user *C.char) *C.char {
    cstr := C.CString("Hello World")
    defer C.free(unsafe.Pointer(cstr))
    return cstr 
}

Мой рубиновый код, как показано ниже

module GoLib
  extend FFI::Library
  ffi_lib './golib.so'

  attach_function :login, [:string], :string
end

GoLib.login("User1") #=> "p\x9A\xA0\xDB\x16V"

Он не возвращается в виде рубиновой строки. Как мне это исправить?

0
Gavin Yap 4 Дек 2018 в 11:10

1 ответ

Лучший ответ

Как указано в комментариях, cstr освобождается после передачи управления Ruby. Вот обходной путь, при котором вы явно управляете указателем в Ruby.

package main

// #include <stdlib.h>
import "C"
import "unsafe"

//export login
func login(user *C.char) *C.char {
    return C.CString("Hello from Go")
}

//export logout
func logout(c *C.char) {
    C.free(unsafe.Pointer(c))
}

func main() {}

В Ruby:

require 'ffi'

module GoLib
  extend FFI::Library
  ffi_lib './so.so'

  attach_function :login, [:string], :strptr
  attach_function :logout, [:pointer], :void
end

s, p = GoLib.login("User1")
puts s
p = FFI::AutoPointer.new(p, GoLib.method(:logout))

Обратите внимание на использование strptr, как показано здесь

3
ssemilla 4 Дек 2018 в 15:55