ls -l 命令的简单实现

思路:先打开目录 -----> 再读取目录 ----->从读到的目录中获取文件名 ----->stat获取文件信息------>显示信息

注意:对于指定的目录,则需要获取它的绝对路径来实现。

/*************************************************************************
    > File Name: ls_v2.c
    > Author: casess
    > Mail: posixssW@gmail.com 
    > Created Time: 2013年12月20日 星期五 14时59分34秒
 ************************************************************************/
//显示目录下的文件信息
//如何显示指定的目录信息:给出那个目录的绝对路径即可
#include<stdio.h>
#include<stdlib.h>
#include<string.h>
#include<unistd.h>
#include<sys/types.h>
#include<dirent.h>
#include<sys/stat.h>
#include<pwd.h>
#include<grp.h>

void do_ls(const char * dirname);
void do_stat(const char * fullpath,const char * filename);
void show_file_info(const char * filename, const struct stat * file_inf_ptr);
char * uid_to_name(uid_t uid);
void mode_to_letters(int mode , char * mode_str);
char * gid_to_name(gid_t uid);

int main(int argc,char * argv[])
{
	if(1 == argc)
		do_ls(".");  //ls current directory
	else
	{
		while(--argc)
		{
			printf("dir : %s \n",* ++argv);
			do_ls( * argv );
		}
	}
	return 0;
}

void do_ls(const char * dirname)
{
	DIR * dir_ptr = NULL;
	struct dirent * dir_entry_ptr = NULL;
	char * fullpath = NULL;
	
	//分配绝对路径名的最大空间
	if( NULL == ( fullpath = (char *)malloc(strlen(*dirname)+1 + NAME_MAX+1) ) )
	{
		puts("malloc in fullpath failed");
		exit(1);
	}

	if(NULL == (dir_ptr = opendir(dirname)))	//打开目录
	{
		fprintf(stderr,"cannot open dir :%s\n",dirname);
		perror("open dir failed :");
		exit(1);
	}
	else
	{
		while(NULL != (dir_entry_ptr = readdir( dir_ptr )) )	//读取目录,成功:返回dirent结构体指针
		{
			sprintf(fullpath,"%s/%s",dirname,dir_entry_ptr->d_name);
		//	printf("%s",fullpath);  //也代表根目录
			do_stat( fullpath , dir_entry_ptr->d_name );
		}
		closedir(dir_ptr);
	}
}

void do_stat(const char * fullpath ,const char * filename)
{
	struct stat file_info;
	if(-1 == stat(fullpath,&file_info))
	{
		perror("get file-info failed ");
		return ;
	}
	else
	{
		show_file_info(filename,&file_info);
	}

}

void show_file_info(const char * filename, const struct stat * file_inf_ptr)
{
	char mode[11];	//file mode
	mode_to_letters(file_inf_ptr->st_mode,mode);

	printf( "%s" , mode );
	printf( " %4d" , (int)file_inf_ptr->st_nlink );
	printf( " %-8s" , uid_to_name(file_inf_ptr->st_uid) );
	printf( " %-8s" , gid_to_name(file_inf_ptr->st_gid) );
	printf( " %8ld" , (long)file_inf_ptr->st_size );
	printf( " %.12s" , 4+ctime(&(file_inf_ptr->st_mtime)) );
	printf( " %s\n" , filename );
}

void mode_to_letters(int mode , char * mode_str)
{
	strncpy(mode_str,"-----------",10);	//此处容易出错,注意mode_str必须要以 ‘\0'结尾且放入的字符串长度不能超过10
	if(S_ISDIR(mode)) mode_str[0] = 'd';	//is directory ?
	if(S_ISCHR(mode)) mode_str[0] = 'c';	//is char devices ?
	if(S_ISBLK(mode)) mode_str[0] = 'b';	//is block devices ?

	if(mode & S_IRUSR) mode_str[1] = 'r';	
	if(mode & S_IWUSR) mode_str[2] = 'w';	
	if(mode & S_IXUSR) mode_str[3] = 'x';
	
	if(mode & S_IRGRP) mode_str[4] = 'r';	
	if(mode & S_IWGRP) mode_str[5] = 'w';	
	if(mode & S_IXGRP) mode_str[6] = 'x';	

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

char * uid_to_name(uid_t uid)
{//更具有可移植性,如果uid不是合法用户,那么就会返回 NULL
	struct passwd * pw_ptr = NULL;
	static char numstr[10];	//static不能省略,否则返回的是栈地址,会出现错误
	if(NULL == (pw_ptr = getpwuid(uid)) )
	{
		sprintf(numstr,"%9d",uid);
		return numstr;
	}
	else
		return pw_ptr->pw_name;
}

char * gid_to_name(gid_t gid)
{//更具有可移植性,如果gid不是合法组,那么就会返回 NULL
	struct group * gr_ptr = NULL;
	static char numstr[10];	//static不能省略,否则返回的是栈地址,会出现错误
	if(NULL == (gr_ptr = getgrgid(gid)) )
	{
		sprintf(numstr,"%9d",gid);
		return numstr;
	}
	else
		return gr_ptr->gr_name;
}



  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值