龙盟编程博客 | 无障碍搜索 | 云盘搜索神器
快速搜索
主页 > 软件开发 > VB开发 >

用VB获得大容量硬盘信息

时间:2009-12-30 15:42来源:未知 作者:admin 点击:
分享到:
这是为什么呢?熟悉Win32API编程的人都知道,在用VB编程时,我们可以用相应的API函数来获得有关系统的硬盘的信息,会不会是错在API的调用函数呢?想到这里我用自编的程序来查看PCA
这是为什么呢?熟悉Win32API编程的人都知道,在用VB编程时,我们可以用相应的API函数来获得有关系统的硬盘的信息,会不会是错在API的调用函数呢?想到这里我用自编的程序来查看PCAlert的有关文件,发现其调用了GetDiskFreeSpace函数,问题就出在这个函数上。下面是该函数的参数说明:ByvallpRootPathNameAsString(为欲查看的分区的根路径如C:),lpSectorsPerClusterAsLong(为一簇的扇区数),lpBytesPerSectorAsLong(为每一扇区的字节数),lpNumberOfFreeClustersAsLong(当前分区中未使用的簇数),lpTotalNumberOfClustersAsLong(总的簇数)AsLong。当我们调用此函数时,是以lpBytesPerSector×lpSectorsPerCluster×TotalNumberOfClusters来计算分区总的大小的,在VB中我用来查看自己的C盘时返回值分别为512、64、65526,因而计算出的C盘的大小只有2.05G。

  以下是关于MSDN中的有关详细说明:对于大于的2G分区,GetDiskFreeSpace函数可能(什么可能,是一定!)返回错误的值,此时函数会屏蔽存在lpNumberOfFreeClusters及lpTotalNumberOfClusters中的值,因此建议不要用该函数来获得大于2G分区的信息。对于大于2G的分区应当使用GetDiskFreeSpaceEx函数(从Win95OEMOSR2开始),此函数可以返回分区的有关正确信息。

  找到了问题所在便可以对症下药了,即用GetDiskFreeSpaceEx函数代替GetDiskFreeSpace函数即可。以下是GetDiskFreeSpaceEx函数中所要传递增的参数

 lpRootPathNameString ,不包括卷名的磁盘根路径名
  lpFreeBytesAvailableToCallerLARGE_INTEGER,指定一个变量,用于容纳调用者可用的字节数量
  lpTotalNumberOfBytesLARGE_INTEGER ,指定一个变量,用于容纳磁盘上的总字节数
  lpTotalNumberOfFreeBytesLARGE_INTEGER,指定一个变量,用于容纳磁盘上可用的字节数

PrivateTypeLARGE_INTEGER
   lowpartAsLong
   highpartAsLong
  EndType

  我们可以看到LARGE_INTEGER是一个由两个long型组成的一个类型,两个long组成表示的都是无符号的数,在转换时应当定义一个single型的变量,使其等于highpart*(2^32-1) lowpart,注意此处的两个long型相当于C/C 中的无符号型整数类型,因为在VB中不存在此种类型,故而在换算时要处理好转换关系。我本人的做法是首先判断long型变量的正负,如是正直接相乘,如是负则用2^32-1减去该值再相乘(具体算法详见下面的代码)。

OptionExplicit
  PrivateTypeLARGE_INTEGER
   lowpartAsLong
   highpartAsLong
  EndType

PrivateDeclareFunctionGetDiskFreeSpaceLib"kernel32"Alias"GetDiskFreeSpaceA"(ByVallpRootPathNameAsString,lpSectorsPerClusterAsLong,lpBytesPerSectorAsLong,lpNumberOfFreeClustersAsLong,lpTotalNumberOfClustersAsLong)AsLong

PrivateDeclareFunctionGetDiskFreeSpaceExLib"kernel32"Alias"GetDiskFreeSpaceExA"_
  (ByVallpRootPathNameAsString,lpFreeBytesAvailableToCallerAsLARGE_INTEGER,_
  lpTotalNumberOfBytesAsLARGE_INTEGER,lpTotalNumberOfFreeBytes_
  AsLARGE_INTEGER)AsLong

PrivateSubCommand1_Click()
   DimlngSectors&
   DimlngTotalCluster&
   DimlngFreeCluster&
   DimlngPerCluster&
   DimlngperBytes&
   DimlngSize#
   GetDiskFreeSpace"c:",lngPerCluster,lngperBytes,lngFreeCluster,lngTotalCluster

 MsgBoxCStr(lngTotalCluster*lngperBytes*lngPerCluster)
   Debug.PrintlngTotalCluster,lngperBytes,lngPerCluster
  EndSub

PrivateTypeLARGE_INTEGER
   lowpartAsLong
   highpartAsLong
  EndType
  PrivateDeclareFunctionGetDiskFreeSpaceLib"kernel32"Alias"GetDiskFreeSpaceA"(ByVallpRootPathNameAsString,lpSectorsPerClusterAsLong,lpBytesPerSectorAsLong,lpNumberOfFreeClustersAsLong,lpTotalNumberOfClustersAsLong)AsLong

PrivateDeclareFunctionGetDiskFreeSpaceExLib"kernel32"Alias"GetDiskFreeSpaceExA"_
  (ByVallpRootPathNameAsString,lpFreeBytesAvailableToCallerAsLARGE_INTEGER,_
  lpTotalNumberOfBytesAsLARGE_INTEGER,lpTotalNumberOfFreeBytes_
  AsLARGE_INTEGER)AsLong

PrivateSubCommand1_Click()
  注释:用GetDiskFreeSpace得到错误的容量
   DimlngSectors&
   DimlngTotalCluster&
   DimlngFreeCluster&
   DimlngPerCluster&
   DimlngperBytes&
   DimlngSize#
   GetDiskFreeSpace"c:",lngPerCluster,lngperBytes,lngFreeCluster,lngTotalCluster

 MsgBoxCStr(lngTotalCluster*lngperBytes*lngPerCluster)
  EndSub

PrivateSubcmdStart_Click()
  注释:用GetDiskFreeSpaceEx得到正确的容量
   DimlngFreeCallerAsLARGE_INTEGER
   DimlngTotalAsLARGE_INTEGER
   DimlngTotalFreeAsLARGE_INTEGER
   DimsngSize#

 GetDiskFreeSpaceEx"c:",lngFreeCaller,lngTotal,lngTotalFree
  注释:以下用来显示出分区总容量(以G为单位)
   MsgBoxGetSize(lngTotal)/2^30
  EndSub

PrivateFunctionGetSize(lngSizeAsLARGE_INTEGER)AsSingle

注释:用来从LARGE_INTEGER型变量中换算出实际的大小
  WithlngSize
   If.highpart<0Then
  GetSize=(2^32-1-.highpart)*(2^32-1)
   Else
  GetSize=.highpart*(2^32-1)
   EndIf
   If.lowpart<0Then
  GetSize=GetSize (2^32-1-.lowpart)
   Else
  GetSize=GetSize .lowpart
   EndIf
  EndWith
  EndFunction

PrivateFunctionGetSize(lngSizeAsLARGE_INTEGER)AsSingle

 注释:用来从LARGE_INTEGER型变量中换算出实际的大小
   WithlngSize
  If.highpart<0Then
   GetSize=(2^32-1-.highpart)*(2^32-1)
  Else
   GetSize=.highpart*(2^32-1)
  EndIf
  If.lowpart<0Then
   GetSize=GetSize (2^32-1-.lowpart)
  Else
   GetSize=GetSize .lowpart
  EndIf
   EndWith
  EndFunction
  

  从LARGE_INTEGER的定义来看,应用此函数理论上可得到2^64/2^30=2^34G的分区的大小,不知这辈子能否用上这么大的硬盘.

->

精彩图集

赞助商链接