如何來(lái)寫自己的ls命令
作者:王姍姍,華清遠(yuǎn)見嵌入式學(xué)院講師。
很多實(shí)際證明,最好的學(xué)習(xí)方法是將相關(guān)的知識(shí)點(diǎn)應(yīng)用到具體的例子中。這樣我們不僅知道了原理,也學(xué)會(huì)了怎么應(yīng)用。在學(xué)習(xí)文件IO時(shí),我們可以嘗試來(lái)寫ls命令。所以在寫ls命令之前,我們必須要明確ls命令能做些什么,然后才能知道要怎么去寫ls命令。
其實(shí)ls的參數(shù)選項(xiàng)很多,大多也可以組合使用。我們必須明確實(shí)現(xiàn)自己的ls命令不是一步就到位的,要先學(xué)會(huì)怎樣去實(shí)現(xiàn)它的基本功能。在這里,我以最簡(jiǎn)單的
ls –l (特定的文件)
作為例子里給大家分析下如何去寫linux的命令。
我們觀察終端的打印信息,怎樣才能按照特定的格式去輸出文件的這些信息?為了完成這個(gè)特定的功能我們要完成以下兩步:
1.如何獲取文件信息
2.如何按格式規(guī)則去輸出文件信息
下面我們來(lái)做進(jìn)一步分析:
第一步,如何來(lái)獲取文件信息。
在C庫(kù)中為我們提供了一組函數(shù)用來(lái)獲取文件(普通文件,目錄,管道,socket,字符,塊)的屬性。
它們的函數(shù)原型
#include
#include
#include
int stat(const char *path, struct stat *buf); /*提供文件名字,獲取文件對(duì)應(yīng)屬性。*/
int fstat(int filedes, struct stat *buf); /*通過(guò)文件描述符獲取文件對(duì)應(yīng)的屬性。*/
int lstat(const char *path, struct stat *buf);/* 連接文件描述命,獲取文件屬性。*/
這里要指出的stat和lstat不同點(diǎn)在于對(duì)于鏈接文件,stat顯示的是鏈接文件指向的實(shí)際的文件的屬性,也就是返回該符號(hào)鏈接引用文件的信息,而lstat顯示的是由返回該符號(hào)鏈接的有關(guān)信息
參數(shù): path:
文件路徑名。 filedes:文件描述詞。
buf:是以下結(jié)構(gòu)體的指針,用來(lái)描述文件對(duì)應(yīng)的屬性
struct stat
{
dev_t st_dev; /* 文件所在設(shè)備的標(biāo)識(shí) */
ino_t st_ino; /* 文件結(jié)點(diǎn)號(hào) */
mode_t st_mode; /* 文件保護(hù)模式 */
nlink_t st_nlink; /* 硬連接數(shù) */
uid_t st_uid; /* 文件用戶標(biāo)識(shí) */
gid_t st_gid; /* 文件用戶組標(biāo)識(shí) */
dev_t st_rdev; /* 文件所表示的特殊設(shè)備文件的設(shè)備標(biāo)識(shí) */
off_t st_size; /* 總大小,字節(jié)為單位 */
blksize_t st_blksize; /* 文件系統(tǒng)的塊大小 */
blkcnt_t st_blocks; /* 分配給文件的塊的數(shù)量,512字節(jié)為單元 */
time_t st_atime; /* 最后訪問(wèn)時(shí)間 */
time_t st_mtime; /* 最后修改時(shí)間 */
time_t st_ctime; /* 最后狀態(tài)改變時(shí)間 */
};
函數(shù)實(shí)現(xiàn)如下:
#include
#include
#include
#include
#include
#include
#include
#include
int main(int argc,char *argv[])
{ if(argc < 2 )
{
printf("commend error!n");
return -1;
}
int i;
struct stat buf;
char out[500];
char *p;
if (lstat(argv[1],&buf)==-1)
{
// printf("No such filen");
return -1;
}
}
這個(gè)時(shí)候我們已經(jīng)得到了buf這個(gè)結(jié)構(gòu)體,從這個(gè)結(jié)構(gòu)體中很容易就能寫出以下信息,實(shí)現(xiàn)起來(lái)比較容易:
//連接數(shù)
printf(" %d",buf.st_nlink);
// 用戶id
struct passwd *user;
user=getpwuid(buf.st_uid);
printf(" %s",user->pw_name);
//組id
struct group *group;
group=getgrgid(buf.st_gid);
printf(" %s",group->gr_name);
//大小
printf(" %d ",buf.st_size);
//時(shí)間
struct tm *t;
t=localtime(&buf.st_ctime);
printf(" %d-%d-%d %d:%d",t->tm_year+1900,
t->tm_mon+1,
t->tm_mday,
t->tm_hour,
t->tm_sec);
printf(" %s",argv[1]);
到目前為止,還剩下文件類型,以及對(duì)于用戶,組,其他用戶的讀寫權(quán)限沒(méi)有解析出來(lái),其實(shí)我們所需要的信息已經(jīng)包含在buf的st_mode中,只要對(duì)這個(gè)返回的st_mode進(jìn)行解析就可以得到我們所需要的信息。
先來(lái)看看文件的類型,如何對(duì)文件類型來(lái)判斷呢?可以使用掩碼來(lái)解碼得到文件的類型。在
#define S_IFMT 0170000
#define S_IFREG 0100000
#define S_IFDIR 0040000
#define S_IFBLK 0060000
....
我們?nèi)绾我袛辔募愋褪欠袷瞧胀ㄎ募?,可以寫如下代碼:
if( (info.st_mode & 0170000) == 0100000)
printf("this is a regular file");
當(dāng)然我們也可采取
#define S_ISFIFO(m) ((m) &(0170000) == (0010000))
#define S_ISREG(m) ( (m) & (0170000) == 0100000))
....
所以我們也可以寫如下代碼:
if( S_ISDIR(info.st_mode) )
printf("this is a regular file");
現(xiàn)在繼續(xù)補(bǔ)充剛剛沒(méi)有寫完的程序:
if (S_ISREG(buf.st_mode)) p="-";
else if (S_ISDIR(buf.st_mode)) p="r";
else if (S_ISCHR(buf.st_mode)) p="c";
else if (S_ISBLK(buf.st_mode)) p="b";
else if (S_ISFIFO(buf.st_mode)) p="f";
else if (S_ISLNK(buf.st_mode)) p="l";
else if (S_ISSOCK(buf.st_mode)) p="s";
還剩下文件對(duì)不同用戶的權(quán)限的描述怎么實(shí)現(xiàn),同理用以上的方式來(lái)實(shí)現(xiàn)。
int n;
for(n=8;n>=0;n--)
{
if(buf.st_mode&(1<
{
case 2: printf("r"); break;
case 1: printf("w"); break;
case 0: printf("x"); break;
}
else
printf("-");
}
這樣將這四段寫在一起,就完成了實(shí)現(xiàn)ls –l 特定的文件的功能。
“本文由華清遠(yuǎn)見http://www.embedu.org/index.htm提供”
華清遠(yuǎn)見