Go 1.6 이상에서 Cgo를 사용하여 함수 포인터를 C 코드에 전달하는 방법은 무엇입니까?

Linda Hamilton
풀어 주다: 2024-10-30 17:46:31
원래의
718명이 탐색했습니다.

How to Pass Function Pointers to C Code Using Cgo in Go 1.6 and Later?

Cgo를 사용하여 C 코드에 함수 포인터 전달

Cgo 함수 포인터 전달 변경 사항

Go 1.6 이상에서는 Cgo에 전달 규칙이 더 엄격해졌습니다. C 코드에 대한 포인터. Go 포인터가 포함된 Go 메모리를 가리키는 Go 포인터를 더 이상 전달할 수 없습니다.

코드 예

다음 Go 코드는 C 코드에 함수 포인터를 전달하는 방법을 보여줍니다.

<code class="go">import (
    "fmt"
    "unsafe"
)

/*
   extern void go_callback_int(void* foo, int p1);

   static inline void CallMyFunction(void* pfoo) {
       go_callback_int(pfoo, 5);
       }
*/
import "C"

//export go_callback_int
func go_callback_int(pfoo unsafe.Pointer, p1 C.int) {
    foo := *(*func(C.int))(pfoo)
    foo(p1)
}

func MyCallback(x C.int) {
    fmt.Println("callback with", x)
}

// we store it in a global variable so that the garbage collector
// doesn't clean up the memory for any temporary variables created.
var MyCallbackFunc = MyCallback

func Example() {
    C.CallMyFunction(unsafe.Pointer(&amp;MyCallbackFunc))
}

func main() {
    Example()
}</code>
로그인 후 복사

오류 및 해결 방법

이 코드를 실행하면 MyCallbackFunc가 가리키는 Go 메모리에 Go 함수 포인터(MyCallback)가 포함되어 있기 때문에 오류가 발생합니다.

To 이 문제를 해결하려면 새로운 규칙을 위반하지 않고 함수 포인터를 C 코드에 전달할 수 있는 방법을 찾아야 합니다.

포인터 대신 ID 사용

한 가지 접근 방식 함수 포인터를 동기화된 데이터 구조에 저장하고 직접 포인터 대신 ID를 C 코드에 전달하는 것입니다. 이렇게 하면 C 코드는 ID를 사용하여 데이터 구조를 통해 함수 포인터에 액세스할 수 있습니다.

ID 기반 함수 포인터 전달 코드

<code class="go">package gocallback

import (
    "fmt"
    "sync"
)

/*
extern void go_callback_int(int foo, int p1);

static inline void CallMyFunction(int foo) {
    go_callback_int(foo, 5);
}
*/
import "C"

//export go_callback_int
func go_callback_int(foo C.int, p1 C.int) {
    fn := lookup(int(foo))
    fn(p1)
}

func MyCallback(x C.int) {
    fmt.Println("callback with", x)
}

func Example() {
    i := register(MyCallback)
    C.CallMyFunction(C.int(i))
    unregister(i)
}

var mu sync.Mutex
var index int
var fns = make(map[int]func(C.int))

func register(fn func(C.int)) int {
    mu.Lock()
    defer mu.Unlock()
    index++
    for fns[index] != nil {
        index++
    }
    fns[index] = fn
    return index
}

func lookup(i int) func(C.int) {
    mu.Lock()
    defer mu.Unlock()
    return fns[i]
}

func unregister(i int) {
    mu.Lock()
    defer mu.Unlock()
    delete(fns, i)
}</code>
로그인 후 복사

위 내용은 Go 1.6 이상에서 Cgo를 사용하여 함수 포인터를 C 코드에 전달하는 방법은 무엇입니까?의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
저자별 최신 기사
인기 튜토리얼
더>
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿
회사 소개 부인 성명 Sitemap
PHP 중국어 웹사이트:공공복지 온라인 PHP 교육,PHP 학습자의 빠른 성장을 도와주세요!