实现linux下的ls

ls的使用

  1. ls -a 列出文件下所有的文件,包括以“.“开头的隐藏文件(linux下文件隐藏文件是以.开头的,如果存在..代表存在着父目录)。
  2. ls -l 列出文件的详细信息,如创建者,创建时间,文件的读写权限列表等等。
  3. ls -F 在每一个文件的末尾加上一个字符说明该文件的类型。"@"表示符号链接、"|"表示FIFOS、"/"表示目录、"="表示套接字。
  4. ls -s 在每个文件的后面打印出文件的大小。 size(大小)
  5. ls -t 按时间进行文件的排序 Time(时间)
  6. ls -A 列出除了"."和".."以外的文件。
  7. ls -R 将目录下所有的子目录的文件都列出来,相当于我们编程中的“递归”实现
  8. ls -L 列出文件的链接名。Link(链接)
  9. ls -S 以文件的大小进行排序

ls的C语言代码实现

#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdlib.h>
#include <errno.h>
#include <dirent.h>
#include <string.h>
#include <linux/limits.h>
#include <pwd.h>
#include <grp.h>
#include <time.h> #define PARAM_NONE 0 //无参数
#define PARAM_A 1 //-a
#define PARAM_L 2 //-l
#define MAXROWLEN 80 //一行最多显示的字符数 int g_leave_len = MAXROWLEN; //一行是剩余长度,用于输出对齐
int g_maxlen; //存放某目录下最长文件名的长度 void my_error(const char* errstring, int line)
{
fprintf(stderr,"line:%d",line);
perror(errstring);
exit(1);
} //打印单个文件,且没有-l参数
void display_single(char *name)
{
int i,len;
//如果本行不足以打印一个文件名则换行
if(g_leave_len < g_maxlen)
{
printf("\n");
g_leave_len = MAXROWLEN;
} len = strlen(name);
len = g_maxlen - len; printf("%-s",name); for(i=0;i<len;i++)
{
printf(" ");
}
printf(" "); g_leave_len = g_leave_len - g_maxlen - 2; } /*获取文件属性并打印*/
void display_attribute(struct stat buf, char *name)
{
char buf_time[32];
struct passwd *psd;
struct group *grp; //获取文件类型
if(S_ISLNK(buf.st_mode))
printf("1");
else if(S_ISREG(buf.st_mode))
printf("-");
else if(S_ISDIR(buf.st_mode))
printf("d");
else if(S_ISCHR(buf.st_mode))
printf("c");
else if(S_ISBLK(buf.st_mode))
printf("b");
else if(S_ISFIFO(buf.st_mode))
printf("f");
else if(S_ISSOCK(buf.st_mode))
printf("s"); //获取文件权限
if(buf.st_mode & S_IRUSR)
printf("r");
else
printf("-");
if(buf.st_mode & S_IWUSR)
printf("w");
else
printf("-");
if(buf.st_mode & S_IXUSR)
printf("x");
else
printf("-"); if(buf.st_mode & S_IRGRP)
printf("r");
else
printf("-");
if(buf.st_mode & S_IWGRP)
printf("w");
else
printf("-");
if(buf.st_mode & S_IXGRP)
printf("x");
else
printf("-"); if(buf.st_mode & S_IROTH)
printf("r");
else
printf("-");
if(buf.st_mode & S_IWOTH)
printf("w");
else
printf("-");
if(buf.st_mode & S_IXOTH)
printf("x");
else
printf("-"); printf(" "); //根据uid和gid获取文件所有者的用户名于组名
psd = getpwuid(buf.st_uid);
grp = getgrgid(buf.st_gid);
printf("%4d",buf.st_nlink);
printf("%-8s",psd->pw_name);
printf("%-9s",grp->gr_name); printf("%6d",buf.st_size);
strcpy(buf_time, ctime(&buf.st_mtime));//将格林位置时间转化成正常时间格式
buf_time[strlen(buf_time) - 1] = 0;
printf(" %s",buf_time);
} //根据flag参数显示文件内容,调用display_single或者display_attribute
void display(int flag,char *pathname)
{
int i,j;
struct stat buf;
char name[NAME_MAX + 1]; for(i=0,j=0;i<strlen(pathname);i++)
{
if(pathname[i] == '/')
{
j = 0;
}
else
name[j++] = pathname[i];
}
name[j] = 0; if(lstat(pathname,&buf) == -1)
{
my_error("stat",__LINE__);
} if(flag == PARAM_NONE)
{
if(name[0] != '.')//不显示隐藏文件
{
display_single(name);
}
}
else if(flag == PARAM_A)
{
display_single(name);
}
else if(flag == PARAM_L)
{
if(name[0] != '.')
{
display_attribute(buf,name);
printf(" %-s\n",name);
}
}
else if(flag == (PARAM_A | PARAM_L))
{
display_attribute(buf,name);
printf(" %-s\n",name);
} } void display_dir(int flag_param,const char *path)
{
DIR* dir;
struct dirent* dirent;
char filenames[256][PATH_MAX+1],temp[PATH_MAX+1];
int count = 0;//总共有多少个文件 if((dir = opendir(path)) == NULL)
{
my_error("opendir",__LINE__);
} //获取文件总数和最长文件名
while((dirent = readdir(dir)) != NULL)
{
if(g_maxlen < strlen(dirent->d_name))
g_maxlen = strlen(dirent->d_name);
count++;
}
closedir(dir); if(count>256)
my_error("文件太多超过了256个",__LINE__); int i,j,len = strlen(path);
//获取目录下所有的文件名
dir = opendir(path);
for(i=0;i<count;i++)
{
dirent = readdir(dir);
if(dirent == NULL)
{
my_error("readdir",__LINE__);
}
strncpy(filenames[i],path,len);
filenames[i][len] = 0;
strcat(filenames[i],dirent->d_name);
filenames[i][len+strlen(dirent->d_name)] = 0;
} //对文件名进行排序
for(i=0;i<count-1;i++)
for(j=i+1;j<count-1;j++)
{
if(strcmp(filenames[i],filenames[j]) > 0)
{
strcpy(temp,filenames[j]);
strcpy(filenames[j] , filenames[i]);
strcpy(filenames[i] , temp);
}
} for(i=0;i<count;i++)
display(flag_param,filenames[i]);
closedir(dir); //没有-l的话打印一个换行符
if((flag_param & PARAM_L) == 0)
printf("\n");
} int main(int argc, char **argv)
{
int i,j,k;
int num;//记录-的个数
char path[PATH_MAX + 1];
char param[32]; // 保存命令行参数
int flag_param = PARAM_NONE;
struct stat buf; j = 0;
num = 0;
for(i=1;i<argc;i++)
{
if(argv[i][0] == '-')
{
for(k=1;k<strlen(argv[i]);k++)
{
param[j] = argv[i][k];
j++;
}
num++;
}
} //现在只支持-a和-l参数
for(i=0;i<j;i++)
{
if(param[i] == 'a')
{
flag_param |= PARAM_A;
}
else if(param[i] == 'l')
{
flag_param |= PARAM_L;
}
else
{
printf("错误的参数:%c\n",param[i]);
exit(1);
}
} param[j] = 0; //如果没有输入文件名或者目录,就显示当前目录
if((num + 1) == argc)
{
strcpy(path,"./");
path[2] = 0;
display_dir(flag_param,path);
return 0;
} i = 1;
for(i=1;i<argc;i++)
{
if(argv[i][0] != '-')
{
strcpy(path,argv[i]);
if(stat(path,&buf) == -1)
my_error("stat",__LINE__);
if(S_ISDIR(buf.st_mode))
{
//判断目录是否以/结尾
if(path[strlen(argv[i]) - 1] != '/')
{
path[strlen(argv[i])] = '/';
path[strlen(argv[i] + 1)] = 0;
}
else
path[strlen(argv[i])] = 0; display_dir(flag_param,path);
}
else
{
display(flag_param,path);
}
}
} return 0;
}
  • 实现截图

实现linux下的ls的更多相关文章

  1. 实现Linux下的ls -l命令

    基本实现了Linux下的ls -l命令,对于不同的文件显示不同的颜色和显示符号链接暂时没有实现: /************************************************** ...

  2. 【Linux基础】linux下修改ls显示的时间格式

    1.修改ls显示格式 ls -l --time-style '+%Y/%m/%d %H:%M:%S' drwxr-x--- edwetl edwetl // :: arc_test ls -l --t ...

  3. linux下出现+ ls --color=auto -l --color=auto...++ echo -ne '\033]0;root@imon-2:~'等

    [root@imon-2 ~]# cd /root/ + cd /root/ ++ echo -ne '\033]0;root@imon-2:~' [root@imon-2 ~]# ll + ls - ...

  4. linux下实现ls()函数遍历目录

    转载请注明原创:http://www.cnblogs.com/StartoverX/p/4600794.html 需求:在linux下遍历目录,输出目录中各文件名. 在linux下遍历目录的相关函数有 ...

  5. [C++]linux下实现ls()函数遍历目录

    转载请注明原创:http://www.cnblogs.com/StartoverX/p/4600794.html 需求:在linux下遍历目录,输出目录中各文件名. 在linux下遍历目录的相关函数有 ...

  6. Linux下执行ls命令提示CMake Error错误

    一.系统环境 Fedora10 二.出错情况 执行ls命令出现如下错误提示: CMake Error: The source directory "/etc/--color=auto&quo ...

  7. linux下的ls命令

    在LINUX系统中有一个重要的概念:一切都是文件.其实这是UNIX哲学的一个体现,而Linux是重写UNIX而来,所以这个概念也就传承了下来.在UNIX系统中,把一切资源都看作是文件,包括硬件设备.U ...

  8. 实现Linux下的ls和ls-l

    ls的C语言代码实现 #include <stdio.h> #include <sys/types.h> #include <sys/stat.h> #includ ...

  9. Linux下用ls和du命令查看文件以及文件夹大小

    ls的用法 ls -l |grep "^-"|wc -l或find ./company -type f | wc -l  查看某文件夹下文件的个数,包括子文件夹里的. ls -lR ...

随机推荐

  1. Java集合----Collection工具类

    Collections 工具类 Collections 是一个操作 Set.List 和 Map 等集合的工具类 Collections 中提供了大量方法对集合元素进行排序.查询和修改等操作,还提供了 ...

  2. GIS-003-在线地图下载及应用

    目前在线地图有谷歌.高德.百度.Bing.soso.天地图.OpenStreet.ArcGIS Online等.在企业应用中,一般需要物理网络隔绝,就有必要下载地图数据之后模拟发布,可以选择下载谷歌影 ...

  3. 批量执行命令:fabric

    Fabric 可以通过 SSH 在多台客户端主机上批量执行任务,是基于 paramiko 封装开发的,paramiko 更底层一些,安装方法如下: [root@localhost ~]$ yum in ...

  4. C语言各种存储模式的区别?最常用的存储模式有哪些?

    DOS用一种段地址结构来编址计算机的内存,每一个物理内存位置都有一个可通过段地址一偏移量的方式来访问的相关地址.为了支持这种段地址结构,大多数C编译程序都允许你用以下6种存储模式来创建程序: ---- ...

  5. OpenGL 4.0 GLSL 基础教程概览——VAO和VBO常用操作接口

    (一) OpenGL  4.3 最新渲染管线图 从OpenGL 2.0 到 OpenGL 3.0变化非常大,但从OpenGL 3.0 到OpenGL 4.0 变化不是太大. 着色器程序直接运行在GPU ...

  6. BNU4204:动物PK

    稀奇稀奇真稀奇,动物园摆出了擂台赛.小动物们纷纷上台比试,谁能获得最后的冠军呢? 动物园长发现小动物们打擂只与自身的三项属性有关:血量,攻击力和防御力.此外,小动物在赛前都为自己准备了一系列的攻击计划 ...

  7. VIM 插入

    不知道有多少VIM新手和我当年(去年)一样,信誓旦旦的以为只有i可以插入 唉,现在想想都觉得可笑,都是Windows下的编辑器用多了的结果 鼠标一点,妈妈再也不用担心我的文本插入了……悲剧! 好了,让 ...

  8. 设计模式之装饰模式(Java实现)

    “怎么了,鱼哥?” “唉,别提了,网购了一件衣服,结果发现和商家描述的差太多了,有色差就算了,质量还不好,质量不好就算了,竟然大小也不行,说好的3个X,邮的却是一个X的,不说了,退货去.你先开讲吧,你 ...

  9. LeetCode——Power of Two

    Description: Given an integer, write a function to determine if it is a power of two. public class S ...

  10. [干货] 有了微信小程序,谁还学ReactNative?

    版权声明:本文由贺嘉原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/145 来源:腾云阁 https://www.qclou ...