dseomy1964 2019-09-20 13:31
浏览 1024
已采纳

golang从dll返回char *作为返回值

I'm using golang to call a Dll function like char* fn(), the dll is not written by myself and I cannot change it. Here's my code:

package main

import (
    "fmt"
    "syscall"
    "unsafe"
)

func main() {
    dll := syscall.MustLoadDLL("my.dll")
    fn := dll.MustFindProc("fn")

    r, _, _ := fn.Call()

    p := (*byte)(unsafe.Pointer(r))
    // define a slice to fill with the p string
    data := make([]byte, 0)

    // loop until find '\0'
    for *p != 0 {
        data = append(data, *p)        // append 1 byte
        r += unsafe.Sizeof(byte(0))    // move r to next byte
        p = (*byte)(unsafe.Pointer(r)) // get the byte value
    }
    name := string(data) // convert to Golang string
    fmt.Println(name)
}

I have some questions:

  1. Is there any better way of doing this? There're hundred of dll functions like this, I'll have to write the loop for all functions.
  2. For very-long-string like 100k+ bytes, will append() cause performance issue?
  3. Solved. the unsafe.Pointer(r) causes linter govet shows warning possible misuse of unsafe.Pointer, but the code runs fine, how to avoid this warning? Solution: This can be solved by adding -unsafeptr=false to govet command line, for vim-ale, add let g:ale_go_govet_options = '-unsafeptr=false'.

enter image description here

  • 写回答

1条回答 默认 最新

  • dongzhan3937 2019-09-21 11:31
    关注

    Casting uintptr as upointer is haram. You must read the rules: https://golang.org/pkg/unsafe/#Pointer

    But there's hacky way, that shouldn't produce warning:

    //go:linkname gostringn     runtime.gostringn
    func gostringn(p uintptr, l int) string
    
    //go:linkname findnull     runtime.findnull
    //go:nosplit
    func findnull(s uintptr) int
    
    // ....
    
    name := gostringn(r, findnull(r))
    

    Functions takes pointer, but we link them from runtime as uintptr because they have same sizeof.

    Might work in theory. But is also frowned upon.

    Getting back to your code, as JimB said, you could do it one line with:

    name := C.GoString((*C.char)(unsafe.Pointer(r)))
    
    本回答被题主选为最佳回答 , 对您是否有帮助呢?
    评论

报告相同问题?

悬赏问题

  • ¥15 关于#matlab#的问题:在模糊控制器中选出线路信息,在simulink中根据线路信息生成速度时间目标曲线(初速度为20m/s,15秒后减为0的速度时间图像)我想问线路信息是什么
  • ¥15 banner广告展示设置多少时间不怎么会消耗用户价值
  • ¥16 mybatis的代理对象无法通过@Autowired装填
  • ¥15 可见光定位matlab仿真
  • ¥15 arduino 四自由度机械臂
  • ¥15 wordpress 产品图片 GIF 没法显示
  • ¥15 求三国群英传pl国战时间的修改方法
  • ¥15 matlab代码代写,需写出详细代码,代价私
  • ¥15 ROS系统搭建请教(跨境电商用途)
  • ¥15 AIC3204的示例代码有吗,想用AIC3204测量血氧,找不到相关的代码。