今天分析下利用 scandir 函數(shù)獲取文件列表。

函數(shù)原型
#include <dirent.h>
int scandir(const char *restrict dirp,
struct dirent ***restrict namelist,
int (*filter)(const struct dirent *),
int (*compar)(const struct dirent **,const struct dirent **));
scandir() 會(huì)掃描目錄 dirp(不包括子目錄),經(jīng)由參數(shù) filter 指定的函數(shù)來(lái)挑選符合條件的目錄結(jié)構(gòu)至參數(shù)namelist 數(shù)組中,最后再調(diào)用參數(shù) compar 指定的函數(shù)來(lái)排序 namelist 數(shù)組中的目錄數(shù)據(jù)。
每次從 dirp 中讀取的目錄結(jié)構(gòu)后都會(huì)傳遞給 filter 進(jìn)行過(guò)濾,若 filter 返回 0 則不會(huì)把該目錄結(jié)構(gòu)復(fù)制到 namelist 數(shù)組中。
若 filter 參數(shù)為 NULL,則選擇所有目錄到 namelist 組中。
scandir() 中會(huì)調(diào)用 qsort() 來(lái)對(duì)獲取的目錄列表進(jìn)行排序,參數(shù) compar 則為 qsort() 的參數(shù),若是要把目錄名稱列表按照字母順序排序則 compar 參數(shù)可使用 alphasort()。
返回值 : 返回獲取到的目錄項(xiàng)的數(shù)量。如果發(fā)生錯(cuò)誤,則返回-1,并設(shè)置errno 以指示錯(cuò)誤。
例子
#include <dirent.h>
#include <stdio.h>
#include <stdlib.h>
int main(void)
{
struct dirent **namelist;
int n;
n = scandir(".", &namelist, NULL, alphasort);
if (n == -1) {
perror("scandir");
exit(EXIT_FAILURE);
}
while (n--) {
printf("%s\n", namelist[n]->d_name);
free(namelist[n]);
}
free(namelist);
exit(EXIT_SUCCESS);
}
運(yùn)行結(jié)果
#./test
tdir
libc.c
libb.c
liba.c
gg.h
..
.
該結(jié)果是按照下標(biāo)倒序顯示的,也可以從下標(biāo) 0 開始顯示,這樣就是按照字母排序的了。
使用 filter 參數(shù)進(jìn)行過(guò)濾
#include <dirent.h>
#include <stdio.h>
#include <stdlib.h>
int myfilter(const struct dirent *entry)
{
return strcmp(entry->d_name, ".") && strcmp(entry->d_name, "..");
}
int main(void)
{
struct dirent **namelist;
int n;
n = scandir(".", &namelist, myfilter, alphasort);
if (n == -1) {
perror("scandir");
exit(EXIT_FAILURE);
}
while (n--) {
printf("%s\n", namelist[n]->d_name);
free(namelist[n]);
}
free(namelist);
exit(EXIT_SUCCESS);
}
運(yùn)行結(jié)果
#./test
tdir
libc.c
libb.c
liba.c
gg.h
獲取以 lib 開頭的文件
int myfilter(const struct dirent *ent)
{
if(ent->d_type != DT_REG)
return 0;
return (strncmp(ent->d_name, "lib", 3) == 0);
}
運(yùn)行結(jié)果如下:
#./test
libc.c
libb.c
liba.c