CGO,如何向C函数传递空参数

10
有时候,C API 需要一个 NULL 指针。
在 CGO 中是否可能实现?
例如,在 Go 语言程序中,我想将 null 参数传递给 strcmp() 函数。
package strutil

/*
#include <stdlib.h>
#include <string.h>
*/
import (
    "C"
    "unsafe"
)

func StrCmp(a, b string) int {
    pca := C.CString(a)
    defer C.free(pca)
    pcb := C.CString(b)
    defer C.free(pcb)
    ret := C.strcmp(pca, pcb)
    return int(ret)
}

如果我将 pca 设置为 nil,会出现此错误:
Exception 0xc0000005 0x0 0x0 0x76828b21
PC=0x76828b21
signal arrived during cgo execution

strutil._Cfunc_strcmp(0x0, 0x761b00, 0x0)
    strutil/_obj/_cgo_gotypes.go:79 +0x49
strutil.StrCmp(0x19, 0x10, 0x4bbf38, 0xa, 0x1fbc1f98, 0x0, 0x0, 0xffff, 0x0,     0x0, ...)

那么,我该如何将NULL传递给strcmp函数呢?
谢谢。

我没有在Go中使用过C API,但是为了允许您的函数接受“Null”,您可以这样做:func StrCmp(a, b *string) int { ... } - Justin Wood
1个回答

14

如果你想将NULL传递给C函数,你可以简单地将nil传递给它。

然而,当你将一个NULL指针发送到函数strcmp(...)时,它的行为没有被文档化。我猜测当你向其传递NULL时,strcmp会失败。最好在传入参数之前检查你的输入并在其中任何一个参数设置为nil时返回错误。

package main

/*
#include <stdlib.h>
#include <string.h>
*/
import "C"

import (
    "errors"
    "fmt"
    "unsafe"
)

func StrCmp(a, b *string) (int, error) {
    if nil == a || nil == b {
        return 0, errors.New("Both strings have to be set")
    }
    pca := C.CString(*a)
    defer C.free(unsafe.Pointer(pca))
    pcb := C.CString(*b)
    defer C.free(unsafe.Pointer(pcb))

    ret := C.strcmp(pca, pcb)
    return int(ret), nil
}

func main() {
    left := "Left"
    right := "Right"
    fmt.Println(StrCmp(&left, &right))
    fmt.Println(StrCmp(&left, &left))
    fmt.Println(StrCmp(nil, &right))
    fmt.Println(StrCmp(&left, nil))
}

如果需要,这是如何将NULL传递给一个接受NULL指针的函数的示例:

package main

/*
#include <stdio.h>
int am_i_null(int* pointer) {
  if (NULL == pointer) {
    return -1;
  }
  return *pointer;
}
*/
import "C"

import (
    "fmt"
    "unsafe"
)

func main() {
    fmt.Println(C.am_i_null(nil))

    var cInteger C.int = 8
    fmt.Println(C.am_i_null(&cInteger))

    var goInteger int = 7
    fmt.Println(C.am_i_null((*C.int)(unsafe.Pointer(&goInteger))))
}

2
C标准的7.1.4.1规定将NULL传递给strcmp会导致未定义的行为(因为strcmp没有记录接受NULL指针)。 - Paul Hankin
谢谢,@Sander,我会尝试你的例子。 - tanbro

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接