荔园在线

荔园之美,在春之萌芽,在夏之绽放,在秋之收获,在冬之沉淀

[回到开始] [上一篇][下一篇]


发信人: oopilix (PADRAD), 信区: Visual
标  题: [ZZ]目录处理函数
发信站: 荔园晨风BBS站 (Mon May 12 21:25:58 2003), 站内信件



目录处理函数
编程语言:c++ builder  作者:王行舟

    在编程时,经常有一些针对目录的操作,如打开目录对话框选择一个目录,直接
创建多级目录,直接删除多级目录,判断某个目录是否存在等。本文就这些问题给出
编程实现方法,并给出详细的程序代码,供各位编程爱好者参考。

一、判断目录是否存在:

  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结构的指针,这个结构较为复杂,成员如下所示:

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可以按照前面介绍的方
法设计;
如果父目录存在,则直接创建目录,否则自我调用创建父目录。


四、直接删除整个目录:

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


--
※ 来源:·荔园晨风BBS站 bbs.szu.edu.cn·[FROM: 61.144.235.41]


[回到开始] [上一篇][下一篇]

荔园在线首页 友情链接:深圳大学 深大招生 荔园晨风BBS S-Term软件 网络书店