C++ Builder中目次处理惩罚的一些函数
副标题#e#
在编程时,常常有一些针对目次的操纵,如打开目次对话框选择一个目次,直接建设多级目次,直接删除多级目次,判定某个目次是否存在等。本文就这些问题给出编程实现要领,并给出具体的措施代码,供列位编程喜好者参考。
一、判定目次是否存在:
C++ Builder中提供了查抄文件是否存在的函数FileExists,但没有提供查抄目次是否存在的函数,我们可以用Windows API函数FindFirstFile实现这个成果。措施实现如下:
设char *Dir为带判定的目次
bool Exist; // 最后功效,暗示目次是否存在
if(Dir[strlen(Dir)]=='\\')Dir[strlen(Dir)-1]='\0'; // 先删除最后的“\”
WIN32_FIND_DATA wfd; // 查找
HANDLE hFind=FindFirstFile(Dir,&wfd);
if(hFind==INVALID_HANDLE_VALUE)Exist=false; // 没有找到配备,目次必定不存在
else
{
if(wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY) // 查抄找到的功效是否目次
Exist=true; // 是目次,目次存在
else
Exist=false; // 是目次,目次不存在
FindClose(hFind);
}
二、打开目次选择对话框选择一个目次:
大多专业软件在要求输入目次的编辑框旁都放了一个按钮,点击后打开一个目次窗口,许多编程喜好者也但愿能把握这个要领。实现这个成果要挪用Windows API函数SHBrowseForFolder,完整声明为WINSHELLAPI LPITEMIDLIST WINAPI SHBrowseForFolder(LPBROWSEINFO lpbi),返回一个ITEMIDLIST范例的指针,通过这个指针挪用函数SHGetPathFromIDList可以确定所选择的目次的全名称。入参为BROWSEINFO布局的指针,这个布局较为巨大,成员如下所示:
#p#副标题#e#
HWND hwndOwner; // 拥有对话框的窗口,可以配置为Application->Handle
LPCITEMIDLIST pidlRoot; // ITEMIDLIST范例的指针,暗示在哪个路径下选择,一般可以配置为NULL
LPSTR pszDisplayName; // 选择后,所选目次的名称(不包括父级目次)被拷贝到这个指针指向的位置
LPCSTR lpszTitle; // 作为标题显示在对话框中目次树的上面,可以按照实际环境配置
UINT ulFlags; // 符号位,有点巨大,一般配置为BIF_RETURNONLYFSDIRS
BFFCALLBACK lpfn; // 回调函数,一般不消,配置为NULL
LPARAM lParam; // 预界说的对话框通报给回调函数的值
int iImage; // 与所选目次相关联的图标在系统图标荟萃中的索引
可以看出,利用函数SHBrowseForFolder还真贫苦,普通喜好者把握它确实有必然的难度,现给出完整措施段如下:
#include <shlobj.h> // 必需包括的头文件
char SelectedDir[MAX_PATH]; // 最终功效
BROWSEINFO bi; // 入参
char FolderName[MAX_PATH]; // 所选目次名称,譬喻选择C:\Windows\Font,则为Font
LPITEMIDLIST ItemID; // 所选目次的系统符号指针
memset(SelectedDir, 0, MAX_PATH); // 初始化最终功效
memset(&bi, 0, sizeof(BROWSEINFO)); // 初始化入参所有数据
bi.hwndOwner = Application->Handle;
bi.pszDisplayName = FolderName;
bi.lpszTitle = "请选择目次"; // 改本钱身但愿的
bi.ulFlags=BIF_RETURNONLYFSDIRS;
ItemID = SHBrowseForFolder(&bi); // 挪用函数,打开目次选择对话框
if(ItemID)
{
SHGetPathFromIDList(ItemID, SelectedDir); // 获取所选目次的全名
GlobalFree(ItemID); // 返回的ItemID占用了系统资源,不要忘了释放
}
三、直接成立多级目次:
Windows API提供了成立目次的函数CreateDirectory,可是挪用前要担保父目次必需存在,不然会失败。其实,有时越级成立多级目次很有用,因为在成立目次出格是成立多层目次时,层层加以判定会大大地增加措施的庞洪水平。如何实现这个成果呢?本人用递归要领设计了一个可以直接成立多级目次的函数,现说明如下,供列位伴侣参考。
bool MakeDirectoryEx(const AnsiString &P) // 入参为规划建设的目次名,按照操纵功效返回"true"或"false"
{
if(P.IsEmpty())return false;
int len=P.Length();
char *Path=P.c_str();
if(Path[len-1]=='\\')
{
len--;
Path[len]='\0';
} // 删除末端的"\"
AnsiString Dir=Path;
// 分隔父目次和自己目次名称
AnsiString Parent;
for(int i=len-1;i>0;i--)
{
if(Dir.IsPathDelimiter(i))
{
Parent=Dir.SubString(0,i);
break;
}
}
if(Parent.IsEmpty())return false; // 目次名称错误
bool Ret=true;
if(Parent.Length()>3) // 假如长度小于3,暗示为磁盘根目次
Ret=DirectoryExistEx(Parent.c_str());// 查抄父目次是否存在
if(!Ret)Ret=MakeDirectoryEx(Parent); // 父目次不存在,递归挪用建设父目次
if(Ret) // 父目次存在,直接建设目次
{
SECURITY_ATTRIBUTES sa;
sa.nLength=sizeof(SECURITY_ATTRIBUTES);
sa.lpSecurityDescriptor=NULL;
sa.bInheritHandle=0;
Ret=CreateDirectory(Path,&sa);
}
return Ret;
}
可以看出根基要领是:
先查抄父目次是否存在,这里用到的函数DirectoryExistEx可以凭据前面先容的要领设计;
假如父目次存在,则直接建设目次,不然自我挪用建设父目次。
四、直接删除整个目次:
#p#分页标题#e#
在DOS下有一个Deltree呼吁,用来删除整个目次,这是一个很有用的成果,惋惜,Windows API提供的函数RemoveDirectory只能删除控目次,就像DOS的RD呼吁一样。编程实现这个成果同样需要递归要领,根基流程是:
查找目次下的所有文件和目次,即挪用API函数FindFirstFile、FindNextFile(*.*)
假如找到文件,则强制删除。所谓强制删除,即删除前先挪用SetFileAttributes把它的属性配置为Normal,然后挪用DeleteFile删除它。
假如找到目次,则举办自我挪用,即开始递归进程。
假如没有找到目次,即暗示为控目次,挪用RemoveDirectory直接删除。
详细措施代码如下:
bool DeleteDirectoryEx(const AnsiString &P)
{
if(P.IsEmpty() || P.Length()<4)return false; // 参数长度必需大于3,即不能为磁盘根目次或空缺
int len=P.Length();
char *Path=P.c_str();
AnsiString Dir=Path;
if(Path[len-1]!='\\')Dir=Dir+'\\';
AnsiString Files=Dir+"*.*";
WIN32_FIND_DATA wfd;
HANDLE hFind=FindFirstFile(Files.c_str(),&wfd);
bool Ret=true;
AnsiString Tmp;
if(hFind!=INVALID_HANDLE_VALUE)
{
bool bFind=true;
while(bFind)
{
if(wfd.cFileName[0]!='.') // . ..
{
Tmp=Dir+wfd.cFileName;
if(wfd.dwFileAttributes & FILE_ATTRIBUTE_DIRECTORY)
{ // 删除所有子目次
Ret=Ret&&DeleteDirectoryEx(Tmp.c_str(),false);
}else
{ // 删除所有文件
SetFileAttributes(Tmp.c_str(),FILE_ATTRIBUTE_NORMAL);
Ret=Ret&&DeleteFile(Tmp.c_str());
}
}
bFind=FindNextFile(hFind,&wfd);
}
FindClose(hFind);
}
if(Ret)return RemoveDirectory(Path);
return false;
}