获取具有未知文件格式的磁盘和/或驱动器的大小,C# .NET Framework?

Posted

技术标签:

【中文标题】获取具有未知文件格式的磁盘和/或驱动器的大小,C# .NET Framework?【英文标题】:Get Size of Disk and/or Drive With Unknown File Format, C# .NET Framework? 【发布时间】:2016-08-30 00:09:51 【问题描述】:

我希望能够以编程方式获取物理磁盘的大小,或作为物理磁盘一部分的逻辑驱动器。

在正常情况下,获取逻辑驱动器大小很容易。但是,对于我正在使用的磁盘,我有意将原始数据写入其中,因此预计它具有未知的驱动器格式

由于 Windows 不知道驱动器格式,Windows 驱动器属性和 .NET Framework 的 DriveInfo 无法告诉我构成此物理磁盘的逻辑驱动器的大小。在 DriveInfo foo 中,格式是指 foo.DriveFormat,foo.IsReady 为 false,而 foo.TotalSize 会触发异常。只需右键单击驱动器并通过 Windows 的属性选项手动获取大小,它显示驱动器大小为 0 字节。

最好使用 C#,使用 .NET Framework 4.0 或更早版本,在 Windows 7 系统上,我如何了解物理磁盘的大小,或组成物理磁盘的逻辑驱动器,如果与它相关联的逻辑驱动器没有已知的文件结构? 具体来说,物理磁盘大小更适合我的目的。

我知道通常有一种使用 Windows 7 的方法,因为我已经看到应用程序可以提供同一磁盘的准确物理磁盘大小。

【问题讨论】:

你见过这个问题吗? ***.com/questions/4008620/getting-disk-size-properly 可能是 IO 控制消息? IOCTL_DISK_GET_LENGTH_INFO / IOCTL_DISK_GET_DRIVE_GEOMETRY @LasseV.Karlsen,我之前发现了这个问题。经过进一步的努力,我发现我没有在 .NET Framework 中正确使用 IOCTL_DISK_GET_LENGTH_INFO(指针和结构的编组使用不正确)。我会相应地更新问题的答案。 【参考方案1】:

解决方案最终相当简单,并且是现有问题 Getting Disk Size Properly 的扩展

在这里发布我的解决方案,因为完整的代码可能会在将来帮助某人: (仅供参考:从http://pinvoke.net 获得并最少修改的 P/Invoke 定义)

[DllImport("kernel32.dll", CharSet = CharSet.Auto, SetLastError = true)]
public static extern IntPtr CreateFile(
    [MarshalAs(UnmanagedType.LPTStr)] string filename,
    [MarshalAs(UnmanagedType.U4)] FileAccess access,
    [MarshalAs(UnmanagedType.U4)] FileShare share,
    IntPtr securityAttributes, // optional SECURITY_ATTRIBUTES struct or IntPtr.Zero
    [MarshalAs(UnmanagedType.U4)] FileMode creationDisposition,
    [MarshalAs(UnmanagedType.U4)] FileAttributes flagsAndAttributes,
    IntPtr templateFile);

[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Auto)]
static extern bool DeviceIoControl(IntPtr hDevice, uint dwIoControlCode,
    IntPtr lpInBuffer, uint nInBufferSize,
    IntPtr lpOutBuffer, uint nOutBufferSize,
    out uint lpBytesReturned, IntPtr lpOverlapped);

[DllImport("kernel32.dll", SetLastError = true)]
static extern bool CloseHandle(IntPtr hObject);

struct GET_LENGTH_INFORMATION

    public long Length;
;
long GetPhysDiskSize(string physDeviceID)

    uint IOCTL_DISK_GET_LENGTH_INFO = 0x0007405C;
    uint dwBytesReturned;

    //Example, physDeviceID == @"\\.\PHYSICALDRIVE1"
    IntPtr hVolume = CreateFile(physDeviceID, FileAccess.ReadWrite,
        FileShare.None, IntPtr.Zero, FileMode.Open, FileAttributes.Normal, IntPtr.Zero);

    GET_LENGTH_INFORMATION outputInfo = new GET_LENGTH_INFORMATION();
    outputInfo.Length = 0;

    IntPtr outBuff = Marshal.AllocHGlobal(Marshal.SizeOf(outputInfo));

    bool devIOPass = DeviceIoControl(hVolume,
                        IOCTL_DISK_GET_LENGTH_INFO,
                        IntPtr.Zero, 0,
                        outBuff, (uint)Marshal.SizeOf(outputInfo),
                        out dwBytesReturned,
                        IntPtr.Zero);

    CloseHandle(hVolume);

    outputInfo = (GET_LENGTH_INFORMATION)Marshal.PtrToStructure(outBuff, typeof(GET_LENGTH_INFORMATION));

    Marshal.FreeHGlobal(hVolume);
    Marshal.FreeHGlobal(outBuff);

    return outputInfo.Length;

【讨论】:

以上是关于获取具有未知文件格式的磁盘和/或驱动器的大小,C# .NET Framework?的主要内容,如果未能解决你的问题,请参考以下文章

C#中怎么获取本地磁盘。我已经实现了把所有的驱动器都获取出来了^但是包括了光驱。

c#怎么获取程序当前运行路径

为啥 exFAT 的“磁盘大小”比 NFTS 的大小要大,因为它们都具有相同的分配单元大小?

在 C# 中获取当前 CPU、RAM 和磁盘驱动器的使用情况

挂载磁盘分区大小

使用 C# 获取文件或流的易读大小(KBMBGBTBKB)