使用CGO封装Windows API
Go使用C的库非常简单,通过cgo这个工具基本上可以说是无缝集成了。下面就演示一下用cgo在Windows下面封装API的过程。注意,请把Go更新到最新一个Weekly版本。
首先,在$GOPATH\src(如果不知道$GOPATH是什么,请移步这里看详细信息)下面新建一个文件夹“w32api”,然后在其内新建一个文件“kernel32.go”,内容如下。
package w32api
// #define WIN32_LEAN_AND_MEAN
// #include <windows.h>
import "C"
import "syscall"
func GetCurrentDirectory() string {
if bufLen := C.GetCurrentDirectoryW(0, nil); bufLen != 0 {
buf := make([]uint16, bufLen)
if bufLen := C.GetCurrentDirectoryW(bufLen, (*C.WCHAR)(&buf[0])); bufLen != 0 {
return syscall.UTF16ToString(buf)
}
}
return ""
}
保存,打开命令行,运行
go build w32api
go install w32api
此时,w32api这个包就编译完成了,用用看吧。
写一个testapp,代码如下。
package main
import "w32api"
func main() {
println(w32api.GetCurrentDirectory())
}
运行之后应该就能看到该文件的当前目录在控制台被打印出来了。感觉如何?是不是简单到令人发指了?这就是为什么Go在很短的时间内就拥有了很多第三方库的秘密,呵呵。
现在重点介绍几个要点,先从kernel32.go的内容说起。
// #define WIN32_LEAN_AND_MEAN// #include <windows.h>
import "C"
这三行应该很熟悉,定义了相关的宏和需要引用的头文件。这里需要注意的是 import “C” 与上一行注释之间不能有空行!否则编译会失败。
之后,就可以用"C.”去引用C库里的函数了,这个前缀还可以引用简单类型,如C.char, C.schar (signed char), C.uchar (unsigned char), C.short, C.ushort (unsigned short), C.int, C.uint (unsigned int), C.long, C.ulong (unsigned long), C.longlong (long long), C.ulonglong (unsigned long long), C.float, C.double。
如果是struct, union和enum的话,需要加上如下前缀,struct_、union_和enum_,比如 C.struct_MSG。
对于字符串的处理比较特殊,cgo提供的字符串处理函数只能处理char类型,这对于Windows上的程序员来说太不够了,因为大多数情况调用的都是Unicode方式的API。我很早之前就提过Bug,且这个Bug一度被标上了Go1的标签,但最近又被从Go1的范畴里剔除了,理由是wchar_t很少见。
没办法了,只能自己先凑活着解决吧!其实也简单,wchar_t其实对应到Go的uint16类型,所以如果要用buffer的话,可以用slice来代替,就像上面代码里写的方法。
buf := make([]uint16, bufLen)
if bufLen := C.GetCurrentDirectoryW(bufLen, (*C.WCHAR)(&buf[0])); bufLen != 0 {
return syscall.UTF16ToString(buf)
}
如果某个函数仅仅只是返回wchar_t指针的话,可以用下面代码得到Go的string。
func UTF16PtrToString(cstr *uint16) string {
if cstr != nil {
us := make([]uint16, 0, 256)
for p := uintptr(unsafe.Pointer(cstr)); ; p += 2 {
u := *(*uint16)(unsafe.Pointer(p))
if u == 0 {
return string(utf16.Decode(us))
}
us = append(us, u)
}
}
return ""
}
另外,cgo目前在windows下面仅支持配合dll使用,还无法做到静态编译*.lib。
以上就是cgo使用的初步介绍,你已经可以开始动手自己玩玩了。也许你更感兴趣的是如何用Go调用C++写的库,恩,好问题,后面我会介绍一种更加简单的封装方式——swig,这个工具大家也许已经知道了,它能自动生成封装层!尽请期待吧!
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· Linux系列:如何用heaptrack跟踪.NET程序的非托管内存泄露
· TypeScript + Deepseek 打造卜卦网站:技术与玄学的结合
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义