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

Get Size of Disk and/or Drive With Unknown File Format, C# .NET Framework?

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

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

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

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

我知道有一种方法通常使用 Windows 7,因为我看到应用程序给出了同一磁盘的准确物理磁盘大小。

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

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

[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;
}