您的位置:首页 > 运维架构 > Linux

UNIX-LINUX编程实践教程->第三章->实例代码注解->ls2

2013-01-29 21:40 821 查看
[b]一 问题[/b]

  对ls1的功能进行扩展,以达到类似ll命令的功能。

[b]二 分析[/b]

  在ls1中,我们利用readdir()函数和dirent结构体来获得目标文件夹内的文件名(dirent->d_name)。
  现在我们借助函数stat()和结构体stat以及上面得到的文件名来获得该目录内文件的详细信息(参看后面的“相关函数与结构体”部分)。

[b]三 实现[/b]

1 头文件

#include<stdio.h>
#include<sys/types.h>
#include<dirent.h>
#include<sys/stat.h>
#include<string.h>


2 相关函数声明

void do_ls(char dirname[]);
void dostat(char *filename);
void show_file_info(char *filename,struct stat *info_p);
void mode_to_letters(int mode,char str[]);
char *uid_to_name(uid_t uid);
char *gid_to_name(gid_t gid);


3 主函数

获得目录的路径,并将该路径传递给do_ls()函数

main(int ac,char *av[])
{
if(ac == 1)
{
do_ls(".");
}
else
{
while(--ac)
{
printf("%s:\n",*++av);
/* 传递目录路径到do_ls函数 */
do_ls(*av);
}
}
}


4 do_ls()函数

结合opendir()函数和readdir()函数,通过目录的路径获得该目录内各文件或文件夹的名称,并将该名称传递给dostat()函数

void do_ls(char dirname[])
{
DIR *dir_ptr;
struct dirent *direntp;
/* 使用opendir()函数“打开”目录,返回指向该目录的指针 */
if((dir_ptr = opendir(dirname))==NULL)
{
fprintf(stderr,"ls2:cannot open %s\n",dirname);
}
else
{
/* 使用readdir()函数读取指针指向的目录,获得dirent结构体 */
while((direntp = readdir(dir_ptr))!=NULL)
{
/* 通过dirent结构体获得文件名,并将文件名传递给dostat()函数 */
dostat(direntp->d_name);
}
closedir(dir_ptr);
}
}


5 dostat()函数

使用stat()函数,通过文件或文件夹的名称获得保存了该文件信息的stat结构体,将该结构体传递给show_file_info()函数

void dostat(char *filename)
{
struct stat info;
/* 使用stat()函数,通过文件名获得stat结构体info */
if(stat(filename,&info) == -1)
{
perror(filename);
}
else
{
/* 将stat结构体传递给show_file_info()函数 */
show_file_info(filename,&info);
}
}


6 show_file_info()函数

将stat结构体中的各项参数分别解析并输出到标准输出

void show_file_info(char *filename,struct stat *info_p)
{
/* 各解析函数声明 */
char *uid_to_name(),*ctime(),*gid_to_name(),*filemode();
void mode_to_letters();
char modestr[11];

/* 解析并输出文件的mode信息 */
mode_to_letters(info_p->st_mode,modestr);
printf("%s",modestr);
/* 文件连接数 */
printf("%4d",(int)info_p->st_nlink);
/* 通过uid获得对应的用户名 */
printf("%-8s",uid_to_name(info_p->st_uid));
/* 通过gid获得对应的组名 */
printf("%-8s",gid_to_name(info_p->st_gid));
/* 文件大小 */
printf("%8ld",(long)info_p->st_size);
/* 最近一次修改时间 */
printf("%.12s",4+ctime(&info_p->st_mtime));
/* 文件名 */
printf("%s\n",filename);
}


7 mode_to_letters()函数

将传递进来的mode参数解析成string格式

void mode_to_letters(int mode,char str[])
{
strcpy(str,"----------");

if(S_ISDIR(mode)) str[0] = 'd';
if(S_ISCHR(mode)) str[0] = 'c';
if(S_ISBLK(mode)) str[0] = 'b';

if(mode&S_IRUSR) str[1] = 'r';
if(mode&S_IWUSR) str[2] = 'w';
if(mode&S_IXUSR) str[3] = 'x';

if(mode&S_IRGRP) str[4] = 'r';
if(mode&S_IWGRP) str[5] = 'w';
if(mode&S_IXGRP) str[6] = 'x';

if(mode&S_IROTH) str[7] = 'r';
if(mode&S_IWOTH) str[8] = 'w';
if(mode&S_IXOTH) str[9] = 'x';
}


其中S_ISDIR、S_ISCHR、S_ISBLK等是宏,定义如下所示:

#define S_ISFIFO(m) (((m)(0170000)) == (0010000))
#define S_ISDIR(m) (((m)(0170000)) == (0040000))
#define S_ISCHR(m) (((m)(0170000)) == (0020000))
#define S_ISBLK(m) (((m)(0170000)) == (0060000))
#define S_ISREG(m) (((m)(0170000)) == (0100000))


8 uid_to_name()函数

通过用户的uid,利用getpwuid()函数,将uid转换为用户名

#include<pwd.h>
char *uid_to_name(uid_t uid)
{
struct passwd *getpwuid(),*pw_ptr;
static char numstr[10];

if((pw_ptr = getpwuid(uid))==NULL)
{
sprintf(numstr,"%d",uid);
return number;
}
else
{
return pw_ptr->pw_name;
}
}


9 gid_to_name()函数

通过用户的gid,利用getgrgid()函数,将gid转换为组名

#include<grp.h>
char *gid_to_name(gid_t gid)
{
struct group *getgrgid(),*grp_ptr;
static char numstr[10];

if((grp_ptr = getgrgid(gid))==NULL)
{
sprintf(numstr,"%d",gid);
return numstr;
}
else
{
return grp_ptr->gr_name;
}
}


相关函数与结构体

1 stat()函数
通过文件名获取文件信息,并保存在stat结构体中
头文件:#include <sys/stat.h>  #include <unistd.h>
函数原型 int stat(const char *file_name, struct stat *buf)
参数    file_name 指向文件名的指针
      buf 指向stat结构体的指针
返回值   0 执行成功
      -1 执行失败,错误代码存于ermo中

2 stat结构体
头文件:#include<sys/stat.h>
struct stat
{
  mode_t st_mode; //文件对应的模式,文件,目录等
  ino_t st_ino; //i-node节点号
  dev_t st_dev; //设备号码
  dev_t st_rdev; //特殊设备号码
  nlink_t st_nlink; //文件的连接数
  uid_t st_uid; //文件所有者
  gid_t st_gid; //文件所有者对应的组
  off_t st_size; //普通文件,对应的文件字节数
  time_t st_atime; //文件最后被访问的时间
  time_t st_mtime; //文件内容最后被修改的时间
  time_t st_ctime; //文件状态(属性)改变时间
  blksize_t st_blksize; //文件内容对应的块大小
  blkcnt_t st_blocks; //文件内容对应的块数量
}

3 ctime()函数
把日期和时间转换为字符串
头文件:#include <cstdio>  #include <ctime>
函数原型 char *ctime(const time_t *time)
参数    time time_t格式的时间
返回值   char * 字符串格式的时间

4 getpwuid()函数
通过用户的uid查找用户的相关信息,并以passwd结构体返回其数据
头文件:#include <sys/types.h>  #include <pwd.h>
函数原型 struct passwd *getpwuid(uid_t uid)
参数    uid 用户的uid
返回值   passwd结构体 uid对应的用户信息
      空指针 出错

5 passwd结构体
struct passwd
{
  char * pw_name; /* Username, POSIX.1 */
  char * pw_passwd; /* Password */
  __uid_t pw_uid; /* User ID, POSIX.1 */
  __gid_t pw_gid; /* Group ID, POSIX.1 */
  char * pw_gecos; /* Real Name or Comment field */
  char * pw_dir; /* Home directory, POSIX.1 */
  char * pw_shell; /* Shell Program, POSIX.1 */
  char *pw_class;
  time_t pw_change;
  time_t pw_expire;
}

6 getgrgid()函数
通过gid,找到该组的相关信息,并以group结构体返回其信息
头文件:#include<grp.h>  #include<sys/types.h>
函数原型 strcut group * getgrgid(gid_t gid)
参数    gid
返回值   group结构体 正常
      NULL 无数据或发生错误

7 group结构体
struct group
{
  char *gr_name; /*组名称*/
  char *gr_passwd; /* 组密码*/
  gid_t gr_gid; /*组识别码*/
  char **gr_mem; /*组成员账号*/
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: