---- 所謂遍歷目錄,就是給定一個目錄,訪問其中的所有文件(包括子目錄下的文件)。迭代是比較常用的遍歷算法。本文利用C++面向對象的特性,通過一個類CBrowseDir,對目錄遍歷進行了封裝。用戶只需掌握該類四個成員函數的用法,就可以在自己的程序中,很方便地實現目錄遍歷。
---- 類CBrowseDir使用了迭代算法。因為算法不是本文重點,筆者不打算展開進一步討論,對其感興趣者可參考相關資料。
一、類成員函數說明:
---- bool SetInitDir(const char *dir);
---- 功能:設置要遍歷的目錄。
---- 參數:dir 指向要遍歷的目錄,可以使用相對路徑,比如"d:..hawk";還可以使用網絡路徑,比如"\wfdhawk"(其中wf是主機名,d是共享目錄,hawk是目錄)。
---- 返回值:返回true,表示設置成功;返回false,說明目錄不可用。
---- bool BeginBrowse(const char *filespec);
---- 功能:開始遍歷目錄中由filespec指定的文件(包括隱藏文件)。
---- 參數:filespec 指定文件類型,可以使用通配符*和?,比如"*.exe"或"a?.*"都是合法參數。注意:filespec中不能包含路徑,象"hawk*.*"是錯誤的。
---- 返回值:函數返回true,表明已順利遍歷完所有文件;返回false,遍歷過程被用戶中止。
---- virtual bool ProcessFile(const char *filename);
---- 功能:虛函數。每找到一個文件,程序就會調用ProcessFile,并把文件名作為參數傳遞給函數。如果函數返回false,則強制遍歷中止,并導致類成員函數函數BeginBrowse返回false。 用戶應該覆寫此函數,以加入自己的處理代碼。
---- 參數:filename 指向一個文件名。注意:filename使用絕對路徑。
---- 返回值:返回true,繼續遍歷;否則,中止遍歷。
---- virtual void ProcessDir (const char *currentdir,const char *parentdir);
---- 功能:虛函數。在遍歷過程中,每進入一個子目錄,程序就會調用ProcessDir,并把目錄名及其上一級目錄名作為參數傳遞給函數。如果該目錄是成員函數SetInitDir指定的初始目錄,則parentdir=NULL。用戶可以覆寫此函數,以加入自己的處理代碼。比如可以在這里統計子目錄的個數。
---- 參數:currentdir 指向一個子目錄。
---- parentdir 指向currentdir的父目錄。
---- 注意:currentdir和parentdir均使用絕對路徑。
二、使用:
---- 把類CBrowseDir的頭文件BrowseDir.h及實現文件BrowseDir.cpp加到項目(Project)中,然后派生自己的類并覆寫虛函數ProcessFile和ProcessDir。遍歷目錄時,先構造一個派生類對象,用成員函數SetInitDir指定目錄,然后調用BeginBrowse開始遍歷。
---- 本文提供了一個例子 example.cpp,它從CBrowseDir派生出子類CStatDir,通過統計函數ProcessFile及ProcessDir的調用次數,可以得知目錄中的文件及子目錄個數。程序都有注釋,這里就不再羅嗦了。
三、注意事項:
---- 1. 類CBrowseDir會改變當前工作目錄。同一個相對路徑,使用CBrowseDir前后,可能會有不同的含義。因此用戶編程時,要小心使用相對路徑。
---- 2. 如果項目(Project)是一個MFC應用程序,直接加入BrowseDir.h及BrowseDir.cpp會導致編譯出錯。這是因為缺省情況下,MFC項目使用了預編譯頭(Precompiled Header),而BrowseDir.h和BrowseDir.cpp是用標準C++語句編寫的,沒用預編譯。一個解決辦法是先用類向導生成類CBrowseDir的"架子",再把相應的代碼拷貝過去。
---- 本文代碼均在Win95、Visual C++ 5.0環境下調試通過。
附源代碼:
/**************************************************
這是CBrowseDir的類定義文件 BrowseDir.h
/**************************************************
#include "stdlib.h"
class CBrowseDir
{
protected:
//存放初始目錄的絕對路徑,以結尾
char m_szInitDir[_MAX_PATH];
public:
//缺省構造器
CBrowseDir();
//設置初始目錄為dir,如果返回false,表示目錄不可用
bool SetInitDir(const char *dir);
//開始遍歷初始目錄及其子目錄下由filespec指定類型的文件
//filespec可以使用通配符 * ?,不能包含路徑。
//如果返回false,表示遍歷過程被用戶中止
bool BeginBrowse(const char *filespec);
protected:
//遍歷目錄dir下由filespec指定的文件
//對于子目錄,采用迭代的方法
//如果返回false,表示中止遍歷文件
bool BrowseDir(const char *dir,const char *filespec);
//函數BrowseDir每找到一個文件,就調用ProcessFile
//并把文件名作為參數傳遞過去
//如果返回false,表示中止遍歷文件
//用戶可以覆寫該函數,加入自己的處理代碼
virtual bool ProcessFile(const char *filename);
//函數BrowseDir每進入一個目錄,就調用ProcessDir
//并把正在處理的目錄名及上一級目錄名作為參數傳遞過去
//如果正在處理的是初始目錄,則parentdir=NULL
//用戶可以覆寫該函數,加入自己的處理代碼
//比如用戶可以在這里統計子目錄的個數
virtual void ProcessDir(const char
*currentdir,const char *parentdir);
};
/*********************************************/
這是CBrowseDir的類實現文件 BrowseDir.cpp
/***********************************************/
#include "stdlib.h"
#include "direct.h"
#include "string.h"
#include "io.h"
#include "browsedir.h"
CBrowseDir::CBrowseDir()
{
//用當前目錄初始化m_szInitDir
getcwd(m_szInitDir,_MAX_PATH);
//如果目錄的最后一個字母不是,則在最后加上一個
int len=strlen(m_szInitDir);
if (m_szInitDir[len-1] != \)
strcat(m_szInitDir,"\");
}
bool CBrowseDir::SetInitDir(const char *dir)
{
//先把dir轉換為絕對路徑
if (_fullpath(m_szInitDir,dir,_MAX_PATH) == NULL)
return false;
//判斷目錄是否存在
if (_chdir(m_szInitDir) != 0)
return false;
//如果目錄的最后一個字母不是,則在最后加上一個
int len=strlen(m_szInitDir);
if (m_szInitDir[len-1] != \)
strcat(m_szInitDir,"\");
return true;
}
bool CBrowseDir::BeginBrowse(const char *filespec)
{
ProcessDir(m_szInitDir,NULL);
return BrowseDir(m_szInitDir,filespec);
}
bool CBrowseDir::BrowseDir
(const char *dir,const char *filespec)
{
_chdir(dir);
//首先查找dir中符合要求的文件
long hFile;
_finddata_t fileinfo;
if ((hFile=_findfirst(filespec,&fileinfo)) != -1)
{
do
{
//檢查是不是目錄
//如果不是,則進行處理
if (!(fileinfo.attrib & _A_SUBDIR))
{
char filename[_MAX_PATH];
strcpy(filename,dir);
strcat(filename,fileinfo.name);
if (!ProcessFile(filename))
return false;
}
} while (_findnext(hFile,&fileinfo) == 0);
_findclose(hFile);
}
//查找dir中的子目錄
//因為在處理dir中的文件時,派生類的ProcessFile有可能改變了
//當前目錄,因此還要重新設置當前目錄為dir。
//執行過_findfirst后,可能系統記錄下了相關信息,因此改變目錄
//對_findnext沒有影響。
_chdir(dir);
if ((hFile=_findfirst("*.*",&fileinfo)) != -1)
{
do
{
//檢查是不是目錄
//如果是,再檢查是不是 . 或 ..
//如果不是,進行迭代
if ((fileinfo.attrib & _A_SUBDIR))
{
if (strcmp(fileinfo.name,".") != 0 && strcmp
(fileinfo.name,"..") != 0)
{
char subdir[_MAX_PATH];
strcpy(subdir,dir);
strcat(subdir,fileinfo.name);
strcat(subdir,"\");
ProcessDir(subdir,dir);
if (!BrowseDir(subdir,filespec))
return false;
}
}
} while (_findnext(hFile,&fileinfo) == 0);
_findclose(hFile);
}
return true;
}
bool CBrowseDir::ProcessFile(const char *filename)
{
return true;
}
void CBrowseDir::ProcessDir(const char
*currentdir,const char *parentdir)
{
}
/*************************************************
這是例子example.cpp
/*************************************************
#include "stdio.h"
#include "BrowseDir.h"
//從CBrowseDir派生出的子類,用來統計目錄中的文件及子目錄個數
class CStatDir:public CBrowseDir
{
protected:
int m_nFileCount; //保存文件個數
int m_nSubdirCount; //保存子目錄個數
public:
//缺省構造器
CStatDir()
{
//初始化數據成員m_nFileCount和m_nSubdirCount
m_nFileCount=m_nSubdirCount=0;
}
//返回文件個數
int GetFileCount()
{
return m_nFileCount;
}
//返回子目錄個數
int GetSubdirCount()
{
//因為進入初始目錄時,也會調用函數ProcessDir,
//所以減1后才是真正的子目錄個數。
return m_nSubdirCount-1;
}
protected:
//覆寫虛函數ProcessFile,每調用一次,文件個數加1
virtual bool ProcessFile(const char *filename)
{
m_nFileCount++;
return CBrowseDir::ProcessFile(filename);
}
//覆寫虛函數ProcessDir,每調用一次,子目錄個數加1
virtual void ProcessDir
(const char *currentdir,const char *parentdir)
{
m_nSubdirCount++;
CBrowseDir::ProcessDir(currentdir,parentdir);
}
};
void main()
{
//獲取目錄名
char buf[256];
printf("請輸入要統計的目錄名:");
gets(buf);
//構造類對象
CStatDir statdir;
//設置要遍歷的目錄
if (!statdir.SetInitDir(buf))
{
puts("目錄不存在。");
return;
}
//開始遍歷
statdir.BeginBrowse("*.*");
//統計結果中,子目錄個數不含 . 及 ..
printf("文件總數: %d 子目錄總數:
%d ",statdir.GetFileCount(),
statdir.GetSubdirCount());
}