FindFirstVolume系列函数遍历驱动器,获取驱动器信息

什么是“卷”?
卷,又称为“逻辑驱动器”,是 NTFS, FAT32 等文件系统组织结构的最高层。卷是存储设备(如硬盘)上由文件系统管理的一块区域,是在逻辑上相互隔离的存储单元。一个磁盘分区至少包含一个卷,当然卷也可以存在于多个磁盘分区上,仅存在于一个分区上的卷称为“简单卷“,存在于多个磁盘分区上的卷称为”多分区卷“或”跨区卷“。最为常见的情况是一个分区只包含 一个卷,一个卷只存在于一个分区上,所以往往会造成的混淆是卷等于分区。卷存在卷标,程序可以通过卷标访问卷。

 

卷挂接在 Unix/Linux/Netware 系统上应用比较常见,Windows 可能由于多用于个人平台,所以知道卷挂接技术的人就比例上来说少了很多。你可以将它理解为:把一个物理设备挂接到一个逻辑盘符或文件夹中来使用,例如:如果需要扩展 C 盘的容量,而 C 盘所在的分区/磁盘已没有空闲空间来扩展它时,可以加装一个 200GB 的新硬盘,然后在 C: 中新增一个目录 ExtDisk,并将这个新磁盘挂接到这个目录中即可,现在 C: 盘的容量就增加了 200GB。有一点要注意,就是卷挂接只能用于 NTFS 系统。

 

4.2.1 遍历卷并获取属性
获取一个主机上的所有驱动器列表有两种方法,一种是使用 GetLogicalDrives 或
GetLogicalDriveStrings,另一种是使用 FindFirstVolume 和 FindNextVolume 组合。
第一种方法获取主机上的逻辑驱动器,也就是所有分配的卷标的驱动器,返回的结果是
驱动器的根路径。第二种方法返回的是“\\?\volume{GUID}”形式的驱动器设备名。
使用 GetDriveType API 可以获取驱动器类型,使用 GetVolumeInformation 可以获取驱
动器属性。
1.关键 API
(1) GetLogicalDrives。
获取主机中所有的逻辑驱动器,以 BitMap 的形式返回,其函数原型如下:
◇返回值
GetLogicalDrive 函数返回一个 DWORD 类型的值,第一位表示所对应的驱动器是否存在。
一般情况下 DWORD 的数据长度是 32 位,在这个 DWORD 中,每一位对应了一个逻辑驱动器是
否存在。第二位如果是“1”则表示驱动器“B:”存在,第 4 位如果是“1”则表示驱动器
“D:”是存在的,以此类推。
(2) GetLogicalDriverStrings。
获取主机中所有驱动器,以驱动器根路径字符串返回,其函数原型如下:
◇参数
nBufferLength:参数 lpBuffer 所指向的内存空间的大小,以字节为单位。
lpBuffer:指向存储返回结果字符串的内存空间。
◇返回值
函数的返回值指明了函数调用是否成功,如果成功则返回缓冲区中返回结果的总长度。
如果返回值大于 nBufferLength,说明给定的缓冲区大小不够,返回值是实际需要的大小。
如果返回 0,则说明函数运行出错。
◇使用说明
这个 API 实现了与 GetLogicalDrives 同样的功能,却以一种更直观的方式返回执行结
果。函数执行结果放在 lpBuffer 所指向的内存区域中,此内存区域大小由 nBufferLength
参数指定,使函数返回结果不至于溢出。在调用此函数前需保证内存分配。
函数调用成功后,将在缓冲区中依次填入本机所具有的驱动器根路径字符串,如在笔者
系统中有 5 个逻辑驱动器“C:\”、“D:\”“E:\”、“F:\”、“I:\”。执行后在缓冲
区中的结果如下:
也就是连续放置了“C:\”、“D:\”“E:\”、“F:\”、“I:\”这 5 个字符串(会
在每个字符串后加一个‘\o’结束符,在所有卷标字符串的最后再加一个结束符)。
(3)FindFirstVolume。
查找主机中的第一个驱动器,返回驱动器设备名,其函数原型如下:
◇参数
lpszVolumeName:指向驱动器名的内存缓冲区。
cchBufferLength:参数 lpszVolumeName 所指向的缓冲区大小,以字节为单位。
◇返回值
驱动器查找句柄, F 猫 NextVolume 和 FindVolumeColse 的参数, 如果执行失败, 返回 NULL。
(4)FindNextVolume
查找主机中后继的逻辑驱动器,其函数原型如下:
◇参数
hFindVolume: FindFirstVolume 所返回的驱动器查找句柄。
lpszVolumeName:指向保存驱动器名的内存缓冲区。
cchBufferLength:参数 lpszVolumeName 所指向的缓冲区大小,以字节为单位。
◇返回值
返回 BOOL 表示是否成功,如果失败说明已经查找完成所有逻辑驱动器。
(5)FindVo1umeClose。
\关闭 FindFirstVolume 打开的卷遍历句柄,其函数原型如下:
◇参数
hFindVolume:要关闭的驱动器查找句柄。
◇返回值
返回 BOOL 值表示是否成功关闭句柄。
(6) GetDriveType。
获取驱动器类型,其函数原型如下:
◇参数
lpRootPathIName:驱动器根路径,如“C:\”。
◇返回值 /
驱动器的类型,如 DRIVE' FIXED 表示硬盘,DRIVE_CDROM 表示光盘等。详见实例 4-2
的 GetDirverInfo 函数。
(7) GetVolumeInformation。
获取逻辑驱动器信息,其函数原型如下:
◇参数
lpRootPathName:输入参数,指向所要获取属性的驱动器的根路径字符串。
lpVolumeNameBuffer:输出参数,返回驱动器名。

 

GetLogicalDriveStrings枚举磁盘,然后可以调用GetDiskFreeSpaceEx获取大小信息。

 

 

GetLogicalDrives

DWORD WINAPI GetLogicalDrives(void);
该函数没有参数,返回值为DWORD,是一个位掩码代表当前的磁盘驱动器。第0位表示驱动器A,第二位表示驱动器B。以此类推,某一位为1表示存在驱动器,为0表示不存在。
 
void GetDisksInformation()
{
	printf("Begin Call GetDisksInformation()\n");
	DWORD dwDisk = GetLogicalDrives();
	int dwMask = 1;
	int step = 1;
	dwMask<<1;
	while (step < 32)
	{
		++step;
		switch (dwMask&dwDisk)
		{
		case 1:
			printf("volume  A\n");
			break;
		case 2:
			printf("volume  B\n");
			break;
		case 4:
			printf("volume  C\n");
			break;
		case 8:
			printf("volume  D\n");
			break;
		case 32:
			printf("volume  E\n");
			break;
		case 64:
			printf("volume  F\n");
			break;
		case 128:
			printf("volume  G\n");
			break;
		case 256:
			printf("volume  H\n");
			break;
		default:
			break;
		}
		dwMask = dwMask<<1;
	}
	printf("end Call GetDisksInformation()\n");
}


 
GetLogicalDriveStrings
DWORD WINAPI GetLogicalDriveStrings(
  _In_   DWORD nBufferLength,
  _Out_  LPTSTR lpBuffer
);
lpBuffer ,存储驱动器根路径字符串的缓冲区内存
nBufferLength 缓冲区 的大小
void GetDisksInformationEx()
{
	printf("Begin Call GetDisksInformationEx()\n");
	CHAR szLogicalDriveString[BUFFERSIZE];
	PCHAR szDrive;

	ZeroMemory(szLogicalDriveString,BUFFERSIZE);
	GetLogicalDriveStrings(BUFFERSIZE - 1, szLogicalDriveString);
	szDrive = szLogicalDriveString;
	while (*szDrive)
	{
		printf("volume %s\n",szDrive);
		szDrive += (lstrlen(szDrive) + 1);
	}
	printf("end Call GetDisksInformationEx()\n");
}


 
 

FindFirstVolume

 
HANDLE WINAPI FindFirstVolume(
  _Out_  LPTSTR lpszVolumeName,      //驱动器名的缓冲区
  _In_   DWORD cchBufferLength       //缓冲区的大小
);
返回值为驱动器的句柄
该函数用来查找主机上第一个驱动器,返回驱动器名。
 
BOOL WINAPI FindNextVolume(
  _In_   HANDLE hFindVolume,    //调用FindFirstVolume返回的驱动器句柄
  _Out_  LPTSTR lpszVolumeName, //驱动器名的缓冲区
  _In_   DWORD cchBufferLength  //缓冲区的大小
);
查找下一个驱动器
 
BOOL WINAPI FindVolumeClose(
  _In_  HANDLE hFindVolume   //调用FindFirstVolume返回的驱动器句柄

);
该函数用来关闭一个驱动器的句柄。被关闭的句柄就不能再在FindNextVolume或者FindVolumeClose中使用了。
 
void FindVolume()
{
	printf("Begin Call FindVolume()\n");
	CHAR  szVolume[MAX_PATH];
	HANDLE hVolume;
	ZeroMemory(szVolume,MAX_PATH);
	hVolume = FindFirstVolume(szVolume,MAX_PATH);
	if (hVolume == INVALID_HANDLE_VALUE)
	{
		printf("Not found the first volume\n");
	}
	printf("volume  %s\n",szVolume);
	while (FindNextVolume(hVolume,szVolume,MAX_PATH))
	{
		printf("volume  %s\n",szVolume);
	}
	FindVolumeClose(hVolume);
	printf("end Call FindVolume()\n");
}

 
 
 
 
 
 
 

#define _WIN32_WINNT 0x0501
#include <windows.h>
#include <stdlib.h>
#include <stdio.h>
#define  BUFSIZE MAX_PATH

BOOL GetDirverInfo(LPSTR szDrive);

int main(void)
{
TCHAR buf[BUFSIZE];
HANDLE hVol;
BOOL bFlag;
/**********************************
* FindFirstVolume(buf,BUFSIZE)
* 功能:查找主机中的第一个驱动器
* 参数:buf,BUFSIZE
* buf 指向驱动器名的内存缓存区
* BUFSIZE 参数buf所指向的缓存区大小
* 返回值 HANDLE :驱动器设备名
**********************************/
hVol = FindFirstVolume(buf,BUFSIZE);

if (hVol == INVALID_HANDLE_VALUE)
{
printf(TEXT("No volmes found!\n"));
return (-1);
}
GetDirverInfo(buf);
while(FindNextVolume(hVol,buf,BUFSIZE))
{
GetDirverInfo(buf);
}
bFlag = FindVolumeClose(hVol);
char ch;
ch = getchar();
return (bFlag);
}

//GetDirverInfo函数

BOOL GetDirverInfo(LPSTR szDrive)
{
UINT uDiveType;
DWORD dwVolumeSerialNumber;
DWORD dwMaximumComponentLength;
DWORD dwFileSystemFlags;
CHAR szFileSystemNameBuffer[BUFSIZE];
CHAR szDirverName[MAX_PATH];
printf("\n%s\n",szDrive);
/*********************
* UINT WINAPI GetDriveType(LPSTR szDrive);
* 功能:判断一个磁盘驱动器的类型
* 参数:LPSTR szDrive 
* 指明要获取属性的驱动器的根目录,如C:\
* 返回值 UINT:表示磁盘类型
**********************/
uDiveType = GetDriveType(szDrive);
switch(uDiveType)
{
case DRIVE_UNKNOWN://未知磁盘类型
printf("The drive type cannot be determined.");
break;
case DRIVE_NO_ROOT_DIR://说明参数是无效
printf("The root path is invalid,for example,no volume is mounted at the path.");
break;
case DRIVE_REMOVABLE://可移动磁盘
printf("The drive is a type that has removable media,for example,\
a floppy drive or removable hard disk.");
break;
case DRIVE_FIXED://固定磁盘
printf("The drive is a type that cannot be removed,for example,a fixed\
hard drive.");
break;
case DRIVE_REMOTE://网络磁盘
printf("The drive is a remote (network) drive");
break;
case DRIVE_CDROM://光驱
printf("The drive is a CD-ROM drive.");
break;
case DRIVE_RAMDISK://RAM
printf("The drive is a RAM disk.");
break;
default:
break;
}
/***********************************
*GetVolumeInformation(szDrive,szDirverName,MAX_PATH,
&dwVolumeSerialNumber,&dwMaximumComponentLength,
&dwFileSystemFlags,szFileSystemNameBuffer,BUFSIZE)
* 功能:获取磁盘驱动器的信息
* 参数:如下
* 返回值:非零即成功
***********************************/
if (!GetVolumeInformation(szDrive,
szDirverName,//卷标(字符串)
MAX_PATH,//szDirveName的长度
&dwVolumeSerialNumber,//用于装载磁盘卷序列号的变量
&dwMaximumComponentLength,//用于装载文件名每一部分的长度
&dwFileSystemFlags,//用于装载一个或多个二进制位标志的变量
szFileSystemNameBuffer,//系统类型
BUFSIZE))
{
return FALSE;
}
if (0!=lstrlen(szDirverName))
{
printf("\nDrive Name is %s\n",szDirverName);
}
printf("\nVolume Serial Number is %u",dwVolumeSerialNumber);
printf("\nMaximun Component Length is %u",dwMaximumComponentLength);
printf("\nSystem Type is %s\n",szFileSystemNameBuffer);
if (dwFileSystemFlags & FILE_SUPPORTS_REPARSE_POINTS)
{
printf("The file system does not support volume mount points.\n");
}
if (dwFileSystemFlags & FILE_VOLUME_QUOTAS)
{
printf("The file system supports disk quotas.\n");
}
if (dwFileSystemFlags & FILE_CASE_SENSITIVE_SEARCH)
{
printf("The file system supports case-sensitive file names.\n");
}
printf("...\n");
return TRUE;
}

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值