• 设为首页
  • 点击收藏
  • 手机版
    手机扫一扫访问
    迪恩网络手机版
  • 关注官方公众号
    微信扫一扫关注
    公众号

Golang下通过syscall调用win32的dll(callingWindowsDLLsfromGo)

原作者: [db:作者] 来自: [db:来源] 收藏 邀请

很多同学比如我虽然很喜欢golang,但是还是需要调用很多遗留项目或者其他优秀的开源项目,这时怎么办呢?我们想到的方法是用package里的syscall结合cgo

注意此处有坑:

在我调试时显示not enough arguments in call to syscall.Syscall

[ `go run dms.go` | done: 260.3744ms ]
# command-line-arguments
.\dms.go:72: not enough arguments in call to syscall.Syscall

exit status 2

因为我参照的是http://golang.org/pkg/syscall/#Syscall ,而其默认的是Linux/Unix的syscall API doc说明,

如何看windows的golang doc呢?

通过godoc command, 调用 godoc -http=:6060
然后在浏览器打开
http://localhost:6060/pkg/syscall/#Syscall,这才是windows的golang pakage api

 

Syscall

func Syscall(trap, nargs, a1, a2, a3 uintptr) (r1, r2 uintptr, err Errno)

 

Implemented in ../runtime/syscall_windows.goc.

Syscall12

func Syscall12(trap, nargs, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11, a12 uintptr) (r1, r2 uintptr, err Errno)

Syscall15

func Syscall15(trap, nargs, a1, a2, a3, a4, a5, a6, a7, a8, a9, a10, a11, a12, a13, a14, a15 uintptr) (r1, r2 uintptr, err Errno)

Syscall6

func Syscall6(trap, nargs, a1, a2, a3, a4, a5, a6 uintptr) (r1, r2 uintptr, err Errno)

Syscall9

func Syscall9(trap, nargs, a1, a2, a3, a4, a5, a6, a7, a8, a9 uintptr) (r1, r2 uintptr, err Errno)

 

例子1:

func test(){
//首先,准备输入参数, GetDiskFreeSpaceEx需要4个参数, 可查MSDN
lpFreeBytesAvailable := int64(0) //注意类型需要跟API的类型相符
lpTotalNumberOfBytes := int64(0)
lpTotalNumberOfFreeBytes := int64(0)


//获取方法的引用
kernel32, _ := syscall.LoadLibrary("Kernel32.dll") // 严格来说需要加上 defer syscall.FreeLibrary(kernel32)
GetDiskFreeSpaceEx, _ := syscall.GetProcAddress(syscall.Handle(kernel32), "GetDiskFreeSpaceExW")


//执行之. 因为有4个参数,故取Syscall6才能放得下. 最后2个参数,自然就是0了
r, _, _ := syscall.Syscall6(uintptr(GetDiskFreeSpaceEx), 4,
            uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr("C:"))),
            uintptr(unsafe.Pointer(&lpFreeBytesAvailable)),
            uintptr(unsafe.Pointer(&lpTotalNumberOfBytes)),
            uintptr(unsafe.Pointer(&lpTotalNumberOfFreeBytes)), 0, 0)
            
// 注意, errno并非error接口的, 不可能是nil
// 而且,根据MSDN的说明,返回值为0就fail, 不为0就是成功
if r != 0 {
    log.Printf("Free %dmb", lpTotalNumberOfFreeBytes/1024/1024)
}
}

例子2,弹出一个简单的对话框messagebox, http://code.google.com/p/go-wiki/wiki/WindowsDLLs

package main

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

func abort(funcname string, err error){
        panic(fmt.Sprintf("%s failed: %v", funcname, err))}var(
        kernel32, _        = syscall.LoadLibrary("kernel32.dll")
        getModuleHandle, _ = syscall.GetProcAddress(kernel32,"GetModuleHandleW")

        user32, _     = syscall.LoadLibrary("user32.dll")
        messageBox, _ = syscall.GetProcAddress(user32,"MessageBoxW"))const(
        MB_OK                =0x00000000
        MB_OKCANCEL          =0x00000001
        MB_ABORTRETRYIGNORE  =0x00000002
        MB_YESNOCANCEL       =0x00000003
        MB_YESNO             =0x00000004
        MB_RETRYCANCEL       =0x00000005
        MB_CANCELTRYCONTINUE =0x00000006
        MB_ICONHAND          =0x00000010
        MB_ICONQUESTION      =0x00000020
        MB_ICONEXCLAMATION   =0x00000030
        MB_ICONASTERISK      =0x00000040
        MB_USERICON          =0x00000080
        MB_ICONWARNING       = MB_ICONEXCLAMATION
        MB_ICONERROR         = MB_ICONHAND
        MB_ICONINFORMATION   = MB_ICONASTERISK
        MB_ICONSTOP          = MB_ICONHAND

        MB_DEFBUTTON1 =0x00000000
        MB_DEFBUTTON2 =0x00000100
        MB_DEFBUTTON3 =0x00000200
        MB_DEFBUTTON4 =0x00000300)

func MessageBox(caption, text string, style uintptr)(result int){
        var nargs uintptr =4
        ret, _, callErr := syscall.Syscall9(uintptr(messageBox),
                nargs,
                0,
                uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(text))),
                uintptr(unsafe.Pointer(syscall.StringToUTF16Ptr(caption))),
                style,
                0,
                0,
                0,
                0,
                0)
        if callErr !=0{
                abort("Call MessageBox", callErr)
        }
        result =int(ret)
        return}

func GetModuleHandle()(handle uintptr){
        var nargs uintptr =0
        if ret, _, callErr := syscall.Syscall(uintptr(getModuleHandle), nargs,0,0,0); callErr !=0{
                abort("Call GetModuleHandle", callErr)
        }else{
                handle = ret
        }
        return}

func main(){
        defer syscall.FreeLibrary(kernel32)
        defer syscall.FreeLibrary(user32)

        fmt.Printf("Return: %d\n",MessageBox("Done Title","This test is Done.", MB_YESNOCANCEL))}

func init(){
        fmt.Print("Starting Up\n")}

 

 

http://golang.org/pkg/syscall/#Syscall 里默认的是Linux/Unix的syscall API doc说明,

 


鲜花

握手

雷人

路过

鸡蛋
该文章已有0人参与评论

请发表评论

全部评论

专题导读
上一篇:
将Go的main包拆分为多个文件发布时间:2022-07-10
下一篇:
graphql GO学习发布时间:2022-07-10
热门推荐
热门话题
阅读排行榜

扫描微信二维码

查看手机版网站

随时了解更新最新资讯

139-2527-9053

在线客服(服务时间 9:00~18:00)

在线QQ客服
地址:深圳市南山区西丽大学城创智工业园
电邮:jeky_zhao#qq.com
移动电话:139-2527-9053

Powered by 互联科技 X3.4© 2001-2213 极客世界.|Sitemap