CGO:如何在 Golang 中使用指针访问 C 中数组中的数据

CGO: How do you use pointers in Golang to access data from an array in C

我正在使用 FFmpeg 为 windows 平台编写一个应用程序,它是 golang 包装器 goav,但我无法理解如何使用 C 指针来访问它们指向的数据数组.

我正在尝试获取存储在 AVFrame 中的数据 class 并使用 Go 将其写入文件,并最终在 OpenGl 中写入纹理以制作具有炫酷过渡的视频播放器。

我认为了解如何转换和访问 C 数据将使编写代码变得容易得多。

我已经删除了 C 代码的所有相关部分,包装器和我的代码,如下所示:

C 代码 - libavutil/frame.h

#include <stdint.h>

typedef struct AVFrame {
#define AV_NUM_DATA_POINTERS 8
    uint8_t *data[AV_NUM_DATA_POINTERS];
}

Golang goav 包装器 - 我真的不知道 unsafe.Pointers 和转换在这里发生了什么,但它让我可以访问底层 C 代码

package avutil

/*
    #cgo pkg-config: libavutil
    #include <libavutil/frame.h>
    #include <stdlib.h>
*/
import "C"
import (
    "unsafe"
)

type Frame C.struct_AVFrame

func AvFrameAlloc() *Frame {
    return (*Frame)(unsafe.Pointer(C.av_frame_alloc()))
}

func Data(f *Frame) *uint8 {
    return (*uint8)(unsafe.Pointer((*C.uint8_t)(unsafe.Pointer(&f.data))))
}

我的 Golang 代码

package main

import "github.com/giorgisio/goav/avutil"

func main() {
    videoFrame := avutil.AvFrameAlloc()

    data := avutil.Data(videoFrame)

    fmt.Println(data) // here i want the values from data[0] to data[7], but how?
}

由于库的作者没有构建切片 header 供您使用,因此您需要将 return 值转换为 unsafe.Pointer,然后再转换为uintptr 这将允许您对其执行指针运算,以便稍后在内存中获取元素。

这里有一些示例代码,应该 运行 as-is 在 go playground 上。

package main

import (
    "fmt"
    "unsafe"
)

func main() {
    nums := []uint8{1, 2, 3, 4, 5, 6, 7, 8}

    val := &nums[0] // val is the equivalent of the *uint8 the Data function returns

    ptr := unsafe.Pointer(val)

    sixthVal := (*uint8)(unsafe.Pointer(uintptr(ptr) + 5*unsafe.Sizeof(*val)))

    fmt.Println("Sixth element:", *sixthVal)
}

当然,你需要非常确定你知道有多少个元素,这样你就不会访问无效内存。