首页 > 后端开发 > Golang > 正文

Go语言实现跨平台获取磁盘空间信息

心靈之曲
发布: 2025-11-10 20:55:00
原创
348人浏览过

Go语言实现跨平台获取磁盘空间信息

本文详细介绍了如何使用go语言在不同操作系统linux/macos和windows)下获取磁盘的可用空间和总空间信息。通过`golang.org/x/sys/unix`和`golang.org/x/sys/windows`包,提供了针对posix和windows系统的具体实现代码,并探讨了如何利用go的构建约束机制来构建一个跨平台的解决方案,帮助开发者便捷地获取类似于`df -h`的磁盘使用报告。

在开发跨平台应用程序时,获取系统磁盘的可用空间和总空间信息是一个常见的需求,类似于在Linux/macOS上执行df -h命令。Go语言本身的标准库并未直接提供一个统一的跨平台API来完成此任务,但通过利用特定操作系统的系统调用包,我们可以实现这一功能。

1. 在POSIX系统(Linux/macOS)上获取磁盘空间

对于基于POSIX标准的系统,如Linux和macOS,我们可以使用golang.org/x/sys/unix包中的Statfs函数来获取文件系统统计信息。这个函数会填充一个Statfs_t结构体,其中包含了我们所需的大部分磁盘信息。

Statfs_t结构体中的关键字段包括:

  • Bsize:文件系统块的大小(字节)。
  • Blocks:文件系统中的总块数。
  • Bfree:文件系统中可用块的总数(包括为超级用户保留的块)。
  • Bavail:非超级用户可用的块数。

通常,我们关注的是非超级用户可用的空间,即Bavail。

立即学习go语言免费学习笔记(深入)”;

以下是一个获取当前工作目录所在文件系统可用空间(字节)的示例:

package main

import (
    "fmt"
    "log"
    "os"

    "golang.org/x/sys/unix"
)

func main() {
    var stat unix.Statfs_t

    // 获取当前工作目录
    wd, err := os.Getwd()
    if err != nil {
        log.Fatalf("获取当前工作目录失败: %v", err)
    }

    // 调用Statfs获取文件系统信息
    err = unix.Statfs(wd, &stat)
    if err != nil {
        log.Fatalf("调用Statfs失败: %v", err)
    }

    // 计算可用空间:非超级用户可用块数 * 每块大小
    availableSpaceBytes := stat.Bavail * uint64(stat.Bsize)
    // 计算总空间:总块数 * 每块大小
    totalSpaceBytes := stat.Blocks * uint64(stat.Bsize)
    // 计算已用空间
    usedSpaceBytes := totalSpaceBytes - (stat.Bfree * uint64(stat.Bsize)) // Bfree 是所有可用块,包含root保留的

    fmt.Printf("路径: %s
", wd)
    fmt.Printf("总空间: %d 字节 (%.2f GB)
", totalSpaceBytes, float64(totalSpaceBytes)/(1024*1024*1024))
    fmt.Printf("可用空间: %d 字节 (%.2f GB)
", availableSpaceBytes, float64(availableSpaceBytes)/(1024*1024*1024))
    fmt.Printf("已用空间: %d 字节 (%.2f GB)
", usedSpaceBytes, float64(usedSpaceBytes)/(1024*1024*1024))
}
登录后复制

注意事项:

ViiTor实时翻译
ViiTor实时翻译

AI实时多语言翻译专家!强大的语音识别、AR翻译功能。

ViiTor实时翻译 116
查看详情 ViiTor实时翻译
  • golang.org/x/sys/unix是一个外部包,需要通过go get golang.org/x/sys/unix进行安装。
  • Bavail是针对非root用户可用的空间,这通常是用户最关心的“可用”空间。

2. 在Windows系统上获取磁盘空间

在Windows平台上,我们需要使用golang.org/x/sys/windows包中的GetDiskFreeSpaceEx函数。这个函数能够提供指定驱动器的可用字节数、总字节数以及总的可用字节数。

GetDiskFreeSpaceEx函数的签名如下: func GetDiskFreeSpaceEx(lpDirectoryName *uint16, lpFreeBytesAvailable *uint64, lpTotalNumberOfBytes *uint64, lpTotalNumberOfFreeBytes *uint64) (err error)

参数说明:

  • lpDirectoryName:指向一个以null结尾的字符串,指定要获取信息的磁盘或UNC路径。例如,"C:"。在Go中,需要将其转换为UTF16指针。
  • lpFreeBytesAvailable:接收调用用户可用的字节数。
  • lpTotalNumberOfBytes:接收磁盘的总字节数。
  • lpTotalNumberOfFreeBytes:接收磁盘上所有用户可用的总字节数。

以下是一个获取C盘磁盘空间信息的示例:

package main

import (
    "fmt"
    "log"
    "syscall"

    "golang.org/x/sys/windows"
)

func main() {
    var freeBytesAvailable uint64
    var totalNumberOfBytes uint64
    var totalNumberOfFreeBytes uint64 // 所有用户可用的总字节数

    // 指定要查询的驱动器路径,并转换为UTF16指针
    drive := "C:\"
    drivePtr, err := syscall.UTF16PtrFromString(drive)
    if err != nil {
        log.Fatalf("转换驱动器路径失败: %v", err)
    }

    // 调用GetDiskFreeSpaceEx
    err = windows.GetDiskFreeSpaceEx(drivePtr,
        &freeBytesAvailable,       // 当前用户可用的字节数
        &totalNumberOfBytes,      // 磁盘总字节数
        &totalNumberOfFreeBytes) // 磁盘上所有用户可用的总字节数

    if err != nil {
        log.Fatalf("调用GetDiskFreeSpaceEx失败: %v", err)
    }

    fmt.Printf("驱动器: %s
", drive)
    fmt.Printf("总空间: %d 字节 (%.2f GB)
", totalNumberOfBytes, float64(totalNumberOfBytes)/(1024*1024*1024))
    fmt.Printf("当前用户可用空间: %d 字节 (%.2f GB)
", freeBytesAvailable, float64(freeBytesAvailable)/(1024*1024*1024))
    fmt.Printf("所有用户可用空间: %d 字节 (%.2f GB)
", totalNumberOfFreeBytes, float64(totalNumberOfFreeBytes)/(1024*1024*1024))
    fmt.Printf("已用空间: %d 字节 (%.2f GB)
", totalNumberOfBytes-totalNumberOfFreeBytes, float64(totalNumberOfBytes-totalNumberOfFreeBytes)/(1024*1024*1024))
}
登录后复制

注意事项:

  • golang.org/x/sys/windows是一个外部包,需要通过go get golang.org/x/sys/windows进行安装。
  • syscall.UTF16PtrFromString用于将Go字符串转换为Windows API所需的UTF16指针。

3. 构建跨平台的磁盘空间获取功能

为了实现一个统一的跨平台API来获取磁盘空间,我们可以利用Go语言的构建约束(Build Constraints)机制。通过在文件名或文件顶部添加特定的注释,Go编译器会根据目标操作系统选择性地编译不同的源文件。

构建约束示例:

  • //go:build windows:仅在Windows系统上编译。
  • //go:build unix:仅在类Unix系统(Linux, macOS, FreeBSD等)上编译。

我们可以创建一个公共接口(例如,一个名为DiskUsage的结构体或函数),然后为每个操作系统提供一个具体的实现文件。

示例文件结构:

disk_usage/
├── disk_usage.go       // 定义公共接口和结构体
├── disk_usage_unix.go  // Unix/Linux/macOS 实现
└── disk_usage_windows.go // Windows 实现
登录后复制

disk_usage.go (公共接口定义):

package disk_usage

import "fmt"

// DiskStatus 存储磁盘使用情况信息
type DiskStatus struct {
    Total      uint64 // 总空间 (字节)
    Free       uint64 // 可用空间 (字节)
    Used       uint64 // 已用空间 (字节)
    FreeForUser uint64 // 对当前用户可用的空间 (字节)
}

// GetDiskUsage 获取指定路径的磁盘使用情况
// 此函数将在不同的操作系统中被具体实现
func GetDiskUsage(path string) (*DiskStatus, error) {
    // 这是一个占位符,实际实现将在特定OS文件中
    return nil, fmt.Errorf("未实现GetDiskUsage函数")
}
登录后复制

disk_usage_unix.go (Unix系统实现):

//go:build unix

package disk_usage

import (
    "fmt"
    "golang.org/x/sys/unix"
)

func GetDiskUsage(path string) (*DiskStatus, error) {
    var stat unix.Statfs_t
    err := unix.Statfs(path, &stat)
    if err != nil {
        return nil, fmt.Errorf("获取Unix磁盘状态失败: %w", err)
    }

    total := stat.Blocks * uint64(stat.Bsize)
    free := stat.Bfree * uint64(stat.Bsize) // 包括root保留的
    freeForUser := stat.Bavail * uint64(stat.Bsize)
    used := total - free // 粗略计算,可能不完全精确,因为文件系统开销

    return &DiskStatus{
        Total:      total,
        Free:       free,
        Used:       used,
        FreeForUser: freeForUser,
    }, nil
}
登录后复制

disk_usage_windows.go (Windows系统实现):

//go:build windows

package disk_usage

import (
    "fmt"
    "syscall"

    "golang.org/x/sys/windows"
)

func GetDiskUsage(path string) (*DiskStatus, error) {
    var freeBytesAvailable uint64
    var totalNumberOfBytes uint64
    var totalNumberOfFreeBytes uint64

    // Windows路径需要以反斜杠结尾,例如 "C:"
    if len(path) > 0 && path[len(path)-1] != '\' {
        path += "\"
    }

    drivePtr, err := syscall.UTF16PtrFromString(path)
    if err != nil {
        return nil, fmt.Errorf("转换Windows驱动器路径失败: %w", err)
    }

    err = windows.GetDiskFreeSpaceEx(drivePtr,
        &freeBytesAvailable,
        &totalNumberOfBytes,
        &totalNumberOfFreeBytes)
    if err != nil {
        return nil, fmt.Errorf("获取Windows磁盘状态失败: %w", err)
    }

    used := totalNumberOfBytes - totalNumberOfFreeBytes

    return &DiskStatus{
        Total:      totalNumberOfBytes,
        Free:       totalNumberOfFreeBytes, // 所有用户可用的总字节数
        Used:       used,
        FreeForUser: freeBytesAvailable,   // 当前用户可用的字节数
    }, nil
}
登录后复制

通过这种方式,当你在不同的操作系统上编译你的Go程序时,Go工具链会自动选择正确的disk_usage_*.go文件进行编译,从而提供一个统一的GetDiskUsage函数接口。

总结

Go语言虽然没有提供一个开箱即用的跨平台API来获取磁盘空间信息,但通过利用其强大的golang.org/x/sys系列包和灵活的构建约束机制,我们可以轻松地为不同操作系统编写特定的实现,并将其封装成一个统一的、易于使用的跨平台功能。这不仅提高了代码的可维护性,也使得Go程序能够更好地适应多样化的运行环境。对于更复杂的磁盘操作,可以考虑使用或参考已有的开源库,例如在问题描述中提到的github.com/ricochet2200/go-disk-usage,它已经实现了这种跨平台逻辑。

以上就是Go语言实现跨平台获取磁盘空间信息的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号