用C语言自己编写一个ls程序


用C语言自己编写一个ls程序

这是自己用C语言编写的一个ls程序,源代码如下:

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

void do_ls(char *);
void dostat(char *);
void show_file_info( char *, struct stat *);
void mode_to_letters( int , char * );
char *uid_to_name( uid_t );
char *gid_to_name( gid_t );

/************关于本文档********************************************
*filename:用C语言自己编写一个ls程序
*purpose:自己编写的一个ls程序
*wrote by: zhoulifa(zhoulifa@163.com) 周立发(http://zhoulifa.bokee.com)
Linux爱好者 Linux知识传播者 SOHO族 开发者 最擅长C语言编程
*date time:2006-07-16 16:00:00
*Note: 任何人可以任意复制代码并运用这些文档,当然包括你的商业用途
* 但请遵循GPL。
*Hope:希望越来越多的人贡献自己的力量,为科学技术发展出力
*********************************************************************/
int main(int argc, char **argv)
{
if ( argc == 1 )
do_ls( "." );
else
while ( --argc ){
printf("%s:/n", *++argv );
chdir(*argv); //切换到指定目录后再显示该目录的内容
do_ls( *argv );
chdir(""); //再回到当前工作目录来
}
return 0;
}

void do_ls( char * dirname )
{
DIR *dir_ptr;
struct dirent *direntp;

if ( ( dir_ptr = opendir( dirname ) ) == NULL ) {/*打开目录*/
fprintf(stderr,"ls1: cannot open %s, not a directory. treat as a file shown below:/n", dirname);
dostat( dirname ); //如果不是目录就当作文件来显示其属性
}
else
{
while ( ( direntp = readdir( dir_ptr ) ) != NULL )
dostat( direntp-
closedir(dir_ptr);
}
}

void dostat( char *filename )
{
struct stat info;

if ( stat(filename, &info) == -1 ) /*取文件信息失败*/
perror( filename );
else /*显示文件信息*/
show_file_info( filename, &info );
}

void show_file_info( char *filename, struct stat *info_p )
{
char modestr[11];

mode_to_letters( info_p-

printf( "%s" , modestr );
printf( "%4d " , (int) info_p-
printf( "%-8s " , uid_to_name(info_p- printf( "%-8s " , gid_to_name(info_p- printf( "%8ld " , (long)info_p-
printf( "%.12s ", 4+ctime(&info_p-
printf( "%s/n" , filename );

}

void mode_to_letters( int mode, char * str )
{
strcpy( str, "----------" ); /* default=无参数 */

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';
}

char *uid_to_name( uid_t uid )
{
struct passwd *pw_ptr;
static char numstr[10];

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

char *gid_to_name( gid_t gid )
{
struct group *grp_ptr;
static char numstr[10];

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

运行结果如下:

./a.out /
/:
drwxr-xr-x 22 root root 4096 Jul 13 12:37 .
drwxr-xr-x 22 root root 4096 Jul 13 12:37 ..
drwxr-xr-x 2 root root 49152 May 21 03:00 lost+found
drwxr-xr-x 130 root root 8192 Jul 16 15:01 etc
drwxr-xr-x 3 root root 4096 May 21 03:01 media
drwxr-xr-x 2 root root 4096 May 21 03:01 cdrom
-rw-r--r-- 1 root root 6773545 Jul 13 08:06 initrd.img
drwxr-xr-x 15 root root 4096 Jun 9 08:40 var
drwxr-xr-x 18 root root 8192 Jul 13 12:37 lib
drwxr-xr-x 11 root root 4096 May 21 03:08 usr
drwxr-xr-x 2 root root 4096 Jul 14 09:41 bin
drwxr-xr-x 3 root root 4096 Jul 13 12:37 boot
drwxr-xr-x 15 root root 15260 Jul 16 14:37 dev
drwxr-xr-x 5 root root 4096 Jun 1 09:22 home
drwxr-xr-x 6 root root 4096 Jul 10 11:30 mnt
dr-xr-xr-x 185 root root 0 Jul 16 22:37 proc
drwxr-xr-x 36 root root 4096 Jul 9 15:45 root
drwxr-xr-x 2 root root 8192 Jul 14 09:44 sbin
drwxrwxrwx 17 root root 4096 Jul 16 16:15 tmp
drwxr-xr-x 10 root root 0 Jul 16 22:37 sys
drwxr-xr-x 2 root root 4096 May 21 03:02 srv
drwxr-xr-x 5 root root 4096 May 29 13:03 opt
drwxr-xr-x 2 root root 4096 May 21 03:02 initrd
-rw-r--r-- 1 root root 1414702 Jul 8 05:22 vmlinuz
drwxr-xr-x 34 administrator administrator 4096 Jul 12 15:31 data

【作者: Liberal】【访问统计:<script language="JavaScript" src="http://counter.blogchina.com/PageServlet?pageid=6508217&blogid=8561"></script>】【2007年10月26日 星期五 18:46】【注册】【打印

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
myls命令包含9个选项: (1) myls : 在缺省选项的情况下,列出当前文件夹下的普通文件(不包含隐藏文件)。 例如:当前目录包含文件home1.c, home2.c, .home3.c,输入myls后,列出的文件名为home1.c, home2.c. (2) myls –a: 列出当前文件夹下的所有文件(包含隐藏文件)。 例如:当前目录包含文件home1.c、home2.c、.home3.c,输入myls -a后,列出所有的文件名为home1.c, home2.c, .home3.c. (3) myls –l: 列出当前文件夹下普通文件的详细信息,包括文件模式,文件链接数,文件所属用户,文件所属用户组,文件大小,文件最后修改时间,文件名。并且在最后一行显示该目录下所显示的文件的文件块数。 例如:显示drwxr-xr-x 3 eli eli 4096 Nov 16 23:07 Desktop (4) myls -R 递归列出文件及其子文件。 例如:当前目录为home, 其中包含文件home1, home2, home3. 其中home1是目录文件,包含文件home11, home12, home2不是目录文件,home3是目录文件,包含文件home31, home32。 输入myls –R后,列出的文件名为 ./home: home1 home2 home3 ./home/home1: home11 home12 ./home/home3: home31 home32. (5) myls –u: 列出当前文件夹下用户x的普通文件,若输入myls -u bb,则显示所属bb的普通文件。 例如:文件home1, home2, home3属于aa,文件tmp1, tmp2, tmp3属于bb, 则若输入myls –u aa, 则显示home1,home2,home3,若输入myls -u bb, 则显示tmp1,tmp2,tmp3。 (6) myls –S: 对文件进行排序,需要输入比较参数。 myls –S 的参数包括: time——按最近修改时间排序 name——按文件名的字典序排序 size——按文件的大小从小到大排序 link——按文件链接数从少到多排序 (7) myls -1: 将当前文件夹下的文件按照一行一个的方式显示。 (8) myls –s: 在各个文件开头显示这个文件的文件块大小。 (9) myls /dirname: 显示/dirname下的文件。 编译 gcc main.c -o myls 执行 ./myls 可加若干参数,具体见上描述 程序并不完整,可能会有BUG,希望广大网友指点,交流~
好的,我可以为您提供一些基本的思路和参考代码。 首先,您需要了解一些基本的 shell 命令和操作符,如 cd、ls、echo、管道符、重定向等。 然后,您可以按照以下步骤编写您的 myshell: 1. 读取用户输入的命令。可以使用 fgets 函数来读取用户输入的命令。 2. 解析用户输入的命令。将用户输入的命令按照空格分割成若干个参数,并判断是否存在管道符或重定向符号等特殊操作符。 3. 根据用户输入的命令,执行相应的操作。可以使用 exec 系列函数来执行命令,使用 fork 函数来创建子进程进行执行。 4. 等待子进程执行完毕,并输出执行结果。 下面是一个简单的 myshell 程序的示例代码: ```c #include <stdio.h> #include <stdlib.h> #include <unistd.h> #include <string.h> #include <sys/wait.h> #define MAX_CMD_LEN 1024 int main() { char cmd[MAX_CMD_LEN]; while (1) { printf("myshell> "); fgets(cmd, MAX_CMD_LEN, stdin); cmd[strlen(cmd) - 1] = '\0'; // 去掉换行符 if (strcmp(cmd, "exit") == 0) { break; } int pid = fork(); if (pid == -1) { perror("fork"); exit(1); } else if (pid == 0) { char *argv[MAX_CMD_LEN], *token; int argc = 0; token = strtok(cmd, " "); while (token != NULL) { argv[argc++] = token; token = strtok(NULL, " "); } argv[argc] = NULL; execvp(argv[0], argv); perror("execvp"); exit(1); } else { int status; waitpid(pid, &status, 0); } } return 0; } ``` 当用户输入 exit 命令时,程序会退出。如果用户输入其他命令,则会创建子进程执行该命令,并等待子进程结束。在子进程中,首先将用户输入的命令解析成若干个参数,然后调用 execvp 函数执行该命令。如果 execvp 函数返回,则说明执行失败,程序会输出错误信息并退出。 希望这个示例代码能够帮到您!

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值