将切片传递给来自c的golang是否会执行内存复制?

时间:2017-04-27 00:15:41

标签: go cgo

我需要将一个大块的float32(或byte)从C传递给go库。

以下是代码:

package main

import (
        "C"
        "fmt"
)

//export PrintInt                                                                                                     
func PrintInt(x []float32) {
        fmt.Println(x)
}

func main() {}

go build -buildmode=c-archive foo.go

编译后

我得到foo.h,其中一部分是:

typedef GoInt64 GoInt;
typedef struct { void *data; GoInt len; GoInt cap; } GoSlice;

#endif

/* End of boilerplate cgo prologue.  */

#ifdef __cplusplus
extern "C" {
#endif


extern void PrintInt(GoSlice p0);

#ifdef __cplusplus
}
#endif

您可以看到GoSlice中的foo.h类型包含指向数据的指针(void *data)。那么PrintIntfoo.h的实现是否会对该数据进行隐式复制?

1 个答案:

答案 0 :(得分:1)

我试图从go侧修改数据,它反映在C.所以它们共享相同的内存。

foo.go:

package main

import (
    "C"
    "fmt"
)

//export PrintInt
func PrintInt(x []float32) {
    fmt.Println(x)
    for i := range x {
        x[i] = -float32(i)
    }
}

func main() {}

foo.c的:

#include <stdio.h>

#include "foo.h"

int main() {
  float bar[32];
  for(int i = 0; i < 32; i++) {
    bar[i] = i;
  }

  GoSlice s;
  s.data = (void*)bar;
  s.len = 32;
  s.cap = 32;
  PrintInt(s);

  for(int i = 0; i < 32; i++) {
    printf("%f ", bar[i]);
  }

  return 0;
}

使用命令:

go build -buildmode=c-archive foo.go
cc foo.c foo.a -o 123 && ./123

我得到了输出:

  

[0 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25   26 27 28 29 30 31]   -0.000000 -1.000000 -2.000000 -6.000000 -4.000000 -5.000000 -6.000000 -7.000000 -8.000000 -9.000000 -10.000000 -11.000000 -12.000000 -13.000000 -14.000000 -15.000000 -16.000000 -17.000000 -18.000000 -19.000000 -20.000000 -21.000000 -22.000000 -23.000000 -24.000000 -25.000000 -26.000000 -27.000000 -28.000000 -29.000000 -30.000000 -31.000000