登录
首页 >  Golang >  Go问答

存在于 Golang 中的 c FILE 类型吗?

来源:stackoverflow

时间:2024-02-19 12:09:26 180浏览 收藏

本篇文章向大家介绍《存在于 Golang 中的 c FILE 类型吗?》,主要包括,具有一定的参考价值,需要的朋友可以参考一下。

问题内容

我最近在尝试将 https://github.com/robpike/strings 编译为 dll(共享库)时遇到了问题,其中 cgo 无法将 os.file 识别为可导出数据类型。我想使用 ctypes 将此 dll 导入到 python 脚本中。

我对 golang 比较陌生,经过大量谷歌搜索后我找不到问题的答案。

strings.go

package main

import (
    "bufio"
    "flag"
    "fmt"
    "io"
    "log"
    "os"
    "strconv"
    "c"
)

var (
    min    = flag.int("min", 6, "minimum length of utf-8 strings printed, in runes")
    max    = flag.int("max", 256, "maximum length of utf-8 strings printed, in runes")
    ascii  = flag.bool("ascii", false, "restrict strings to ascii")
    offset = flag.bool("offset", false, "show file name and offset of start of each string")
)

var stdout *bufio.writer

func main() {
}

//export do
func do(file *os.file) {
    in := bufio.newreader(file)
    str := make([]rune, 0, *max)
    filepos := int64(0)
    print := func() {
        if len(str) >= *min {
            s := string(str)
            // if *offset {
                // fmt.printf("%s:#%d:\t%s\n", name, filepos-int64(len(s)), s)
            // } else {
            fmt.println(s)
            // }
        }
        str = str[0:0]
    }
    for {
        var (
            r   rune
            wid int
            err error
        )
        // one string per loop.
        for ; ; filepos += int64(wid) {
            r, wid, err = in.readrune()
            if err != nil {
                if err != io.eof {
                    log.print(err)
                }
                return
            }
            if !strconv.isprint(r) || *ascii && r >= 0xff {
                print()
                continue
            }
            // it's printable. keep it.
            if len(str) >= *max {
                print()
            }
            str = append(str, r)
        }
    }
}

期望go build -o strings.dll -buildmode=c-shared 输出 strings.dll 以及导出函数 do

问题.\strings.go:41:15:导出中不支持 go 类型:os.file

编辑:我遇到了这个问题:read 文件描述符:句柄无效。

python 驱动程序(windows amd64 上为 3.10.5):

from ctypes import *
mydll = cdll.LoadLibrary("./strings.dll")

strings = mydll.dodo
strings.argtypes = [c_int]
strings.restype = c_void_p

with open("./go.mod", "rb") as h:
    print(strings(h.fileno()))

正确答案


cgo 不支持将结构映射到 c。实际上,您在导出函数中只能使用字符串、整数和原生 c 类型。

调用函数可以传递文件描述符,而不是 os.file。它是一个整数。

我将 func dodo(fd int) 添加到您的代码中,并导出此函数而不是 do

package main

import (
    "c"
    "bufio"
    "flag"
    "fmt"
    "io"
    "log"
    "os"
    "strconv"
)

var (
    min    = flag.int("min", 6, "minimum length of utf-8 strings printed, in runes")
    max    = flag.int("max", 256, "maximum length of utf-8 strings printed, in runes")
    ascii  = flag.bool("ascii", false, "restrict strings to ascii")
    offset = flag.bool("offset", false, "show file name and offset of start of each string")
)

var stdout *bufio.writer

func main() {
}

//export dodo
func dodo(fd int) {
    file := os.newfile(uintptr(fd), "file descriptor")
    if file == nil {
        return
    }
    do(file)
}
func do(file *os.file) {
    in := bufio.newreader(file)
    str := make([]rune, 0, *max)
    filepos := int64(0)
    print := func() {
        if len(str) >= *min {
            s := string(str)
            // if *offset {
            // fmt.printf("%s:#%d:\t%s\n", name, filepos-int64(len(s)), s)
            // } else {
            fmt.println(s)
            // }
        }
        str = str[0:0]
    }
    for {
        var (
            r   rune
            wid int
            err error
        )
        // one string per loop.
        for ; ; filepos += int64(wid) {
            r, wid, err = in.readrune()
            if err != nil {
                if err != io.eof {
                    log.print(err)
                }
                return
            }
            if !strconv.isprint(r) || *ascii && r >= 0xff {
                print()
                continue
            }
            // it's printable. keep it.
            if len(str) >= *max {
                print()
            }
            str = append(str, r)
        }
    }
}

构建代码(linux):

go build -o libstrings.so -buildmode=c-shared .

此命令生成 libstrings.hlibstrings.so

使用 libstrings.so 的示例 some.c 程序

// perror
#include 
// open and o_rdonly
#include
// close
#include 
// dodo
#include "libstrings.h"

int main(int argc, char **argv) {
    int fd = open("go.mod", o_rdonly);
    if (fd < 0) {
        perror("failed to open file");
        return 1;
    }
    dodo(fd);
    close(fd);
    return 0;
}

构建:

gcc -o some.exe some.c -l ./ -lstrings

执行:

ld_library_path=. ./some.exe

结果:它按预期打印 go.mod 文件的内容。

module example.org
go 1.19

终于介绍完啦!小伙伴们,这篇关于《存在于 Golang 中的 c FILE 类型吗?》的介绍应该让你收获多多了吧!欢迎大家收藏或分享给更多需要学习的朋友吧~golang学习网公众号也会发布Golang相关知识,快来关注吧!

声明:本文转载于:stackoverflow 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>