0

0

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

心靈之曲

心靈之曲

发布时间:2025-11-10 20:55:00

|

378人浏览过

|

来源于php中文网

原创

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\n", wd)
    fmt.Printf("总空间: %d 字节 (%.2f GB)\n", totalSpaceBytes, float64(totalSpaceBytes)/(1024*1024*1024))
    fmt.Printf("可用空间: %d 字节 (%.2f GB)\n", availableSpaceBytes, float64(availableSpaceBytes)/(1024*1024*1024))
    fmt.Printf("已用空间: %d 字节 (%.2f GB)\n", usedSpaceBytes, float64(usedSpaceBytes)/(1024*1024*1024))
}

注意事项:

sematic
sematic

一个开源的机器学习平台

下载
  • 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\n", drive)
    fmt.Printf("总空间: %d 字节 (%.2f GB)\n", totalNumberOfBytes, float64(totalNumberOfBytes)/(1024*1024*1024))
    fmt.Printf("当前用户可用空间: %d 字节 (%.2f GB)\n", freeBytesAvailable, float64(freeBytesAvailable)/(1024*1024*1024))
    fmt.Printf("所有用户可用空间: %d 字节 (%.2f GB)\n", totalNumberOfFreeBytes, float64(totalNumberOfFreeBytes)/(1024*1024*1024))
    fmt.Printf("已用空间: %d 字节 (%.2f GB)\n", 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,它已经实现了这种跨平台逻辑。

相关专题

更多
golang如何定义变量
golang如何定义变量

golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

174

2024.02.23

golang有哪些数据转换方法
golang有哪些数据转换方法

golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

224

2024.02.23

golang常用库有哪些
golang常用库有哪些

golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

335

2024.02.23

golang和python的区别是什么
golang和python的区别是什么

golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

206

2024.03.05

golang是免费的吗
golang是免费的吗

golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

388

2024.05.21

golang结构体相关大全
golang结构体相关大全

本专题整合了golang结构体相关大全,想了解更多内容,请阅读专题下面的文章。

193

2025.06.09

golang相关判断方法
golang相关判断方法

本专题整合了golang相关判断方法,想了解更详细的相关内容,请阅读下面的文章。

188

2025.06.10

golang数组使用方法
golang数组使用方法

本专题整合了golang数组用法,想了解更多的相关内容,请阅读专题下面的文章。

191

2025.06.17

vlookup函数使用大全
vlookup函数使用大全

本专题整合了vlookup函数相关 教程,阅读专题下面的文章了解更多详细内容。

28

2025.12.30

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.3万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

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