Ottieni la dimensione del file sul disco

var length = new System.IO.FileInfo(path).Length; 

Questo dà la dimensione logica del file, non la dimensione sul disco.

Desidero ottenere la dimensione di un file sul disco in C # (preferibilmente senza interop ) come verrebbe segnalato da Windows Explorer.

Dovrebbe dare la dimensione corretta, anche per:

  • Un file compresso
  • Un file sparse
  • Un file frammentato

Questo utilizza GetCompressedFileSize, come suggerito da ho1, così come GetDiskFreeSpace, come suggerito da PaulStack, tuttavia utilizza P / Invoke. L’ho provato solo per i file compressi e ho il sospetto che non funzioni per i file frammentati.

  public static long GetFileSizeOnDisk(string file) { FileInfo info = new FileInfo(file); uint dummy, sectorsPerCluster, bytesPerSector; int result = GetDiskFreeSpaceW(info.Directory.Root.FullName, out sectorsPerCluster, out bytesPerSector, out dummy, out dummy); if (result == 0) throw new Win32Exception(); uint clusterSize = sectorsPerCluster * bytesPerSector; uint hosize; uint losize = GetCompressedFileSizeW(file, out hosize); long size; size = (long)hosize << 32 | losize; return ((size + clusterSize - 1) / clusterSize) * clusterSize; } [DllImport("kernel32.dll")] static extern uint GetCompressedFileSizeW([In, MarshalAs(UnmanagedType.LPWStr)] string lpFileName, [Out, MarshalAs(UnmanagedType.U4)] out uint lpFileSizeHigh); [DllImport("kernel32.dll", SetLastError = true, PreserveSig = true)] static extern int GetDiskFreeSpaceW([In, MarshalAs(UnmanagedType.LPWStr)] string lpRootPathName, out uint lpSectorsPerCluster, out uint lpBytesPerSector, out uint lpNumberOfFreeClusters, out uint lpTotalNumberOfClusters); 

Il codice riportato sopra non funziona correttamente su Windows Server 2008 o 2008 R2 o Windows 7 e Windows Vista, poiché la dimensione del cluster è sempre zero (GetDiskFreeSpaceW e GetDiskFreeSpace restituiscono -1 anche con UAC disabilitato.) Ecco il codice modificato che funziona.

C #

 public static long GetFileSizeOnDisk(string file) { FileInfo info = new FileInfo(file); uint clusterSize; using(var searcher = new ManagementObjectSearcher("select BlockSize,NumberOfBlocks from Win32_Volume WHERE DriveLetter = '" + info.Directory.Root.FullName.TrimEnd('\\') + "'") { clusterSize = (uint)(((ManagementObject)(searcher.Get().First()))["BlockSize"]); } uint hosize; uint losize = GetCompressedFileSizeW(file, out hosize); long size; size = (long)hosize << 32 | losize; return ((size + clusterSize - 1) / clusterSize) * clusterSize; } [DllImport("kernel32.dll")] static extern uint GetCompressedFileSizeW( [In, MarshalAs(UnmanagedType.LPWStr)] string lpFileName, [Out, MarshalAs(UnmanagedType.U4)] out uint lpFileSizeHigh); 

VB.NET

  Private Function GetFileSizeOnDisk(file As String) As Decimal Dim info As New FileInfo(file) Dim blockSize As UInt64 = 0 Dim clusterSize As UInteger Dim searcher As New ManagementObjectSearcher( _ "select BlockSize,NumberOfBlocks from Win32_Volume WHERE DriveLetter = '" + _ info.Directory.Root.FullName.TrimEnd("\") + _ "'") For Each vi As ManagementObject In searcher.[Get]() blockSize = vi("BlockSize") Exit For Next searcher.Dispose() clusterSize = blockSize Dim hosize As UInteger Dim losize As UInteger = GetCompressedFileSizeW(file, hosize) Dim size As Long size = CLng(hosize) << 32 Or losize Dim bytes As Decimal = ((size + clusterSize - 1) / clusterSize) * clusterSize Return CDec(bytes) / 1024 End Function  _ Private Shared Function GetCompressedFileSizeW( _ <[In](), MarshalAs(UnmanagedType.LPWStr)> lpFileName As String, _  lpFileSizeHigh As UInteger) _ As UInteger End Function 

Secondo i forum social di MSDN:

La dimensione su disco deve essere la sum delle dimensioni dei cluster che memorizzano il file:
http://social.msdn.microsoft.com/Forums/en-US/Vsexpressvcs/thread/85bf76ac-a254-41d4-a3d7-e7803c8d9bc3
Dovrai immergerti in P / Invoke per trovare la dimensione del cluster; GetDiskFreeSpace () lo restituisce.

Vedi Come ottenere le dimensioni sul disco di un file in C # .

Ma si prega di notare il punto che questo non funzionerà in NTFS in cui è triggersta la compressione.

Penso che sarà così:

 double ifileLength = (finfo.Length / 1048576); //return file size in MB .... 

Sto ancora facendo alcuni test per questo, per ottenere una conferma.