用C++实现Linux中shell的ls功能
实现输出当前目录下的文件名
ls功能:
方法一:
#include <iostream>
#include <algorithm>
#include <stdio.h>
#include <stdlib.h>
#include <dirent.h>
#include <sys/types.h>
#include <string.h>
#include <string>
using namespace std; bool cmp( string s1,string s2){
return s1<s2;
} int main()
{
DIR *dir;
char s[];
string data[];
int tot=;
struct dirent *rent;
dir =opendir(".");
while(rent=readdir(dir))
{
strcpy(s,rent->d_name);
if(s[]!='.'){
data[tot]=s;
tot++;
}
}
sort(data,data+tot,cmp);
for(int i=;i<tot;i++)
cout<<data[i]<<" ";
puts("");
closedir(dir);
return ;
}
方法二:
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include <pwd.h>
#include <grp.h>
#include <time.h>
#include <dirent.h>
int do_ls(char *dir,char *filename,int lflag)
{
int n;
struct stat buf;
char out[];
struct passwd *pw;
struct group *gr;
struct tm *t;
if(lflag == ) //如果不带l参数,直接显示文件/目录名
{
printf("%s\t",filename);
return ;
}
if(lstat(dir,&buf)<)
{
fprintf(stderr,"stat error:%s\n",strerror(errno));
return -;
}
switch(buf.st_mode & S_IFMT) //获取字符串的属性:普通文件-、目录d、字符设备c、块设备b、管道文件p、连接文件l、套接字文件s
{
case S_IFREG:
printf("-");
break;
case S_IFDIR:
printf("d");
break;
case S_IFCHR:
printf("c");
break;
case S_IFBLK:
printf("b");
break;
case S_IFIFO:
printf("p");
break;
case S_IFLNK:
printf("l");
break;
case S_IFSOCK:
printf("s");
break;
}
for(n=; n>=; n--) //打印文件的读写属性:读r、写w、执行x、无权限-
{
if(buf.st_mode&(<<n))
{
switch(n%)
{
case :
printf("r");
break;
case :
printf("w");
break;
case :
printf("x");
break;
default:
break;
}
}
else
{
printf("-");
}
}
printf(" %d",buf.st_nlink); //硬链接数,此链接非彼链接,指(包含)目录的个数,文件为1,目录起始为2,再加上目录里包含的目录个数(不递归,只一层)
pw = getpwuid(buf.st_uid); //所属用户名
printf(" %s",pw->pw_name);
gr = getgrgid(buf.st_gid); //所属组名
printf(" %s",gr->gr_name);
printf(" %ld",buf.st_size); //字节计总大小
t = localtime(&buf.st_atime); //最后一次访问时间
printf(" %d-%d-%d %d:%d"
,t->tm_year+
,t->tm_mon+
,t->tm_mday
,t->tm_hour
,t->tm_min);
printf(" %s ",filename);
if(S_ISLNK(buf.st_mode)) //判断是否为链接,是返回真
{
printf(" -> ");
if(readlink(filename,out,)==-)
{
//printf("readlink error\n");
}
printf("%s",out);
}
printf("\n");
return ;
}
int ls_prepare(char *w,int aflag,int lflag) //ls的准备工作
{
struct stat buf; //man lstat可以看到此结构
char name[];
DIR *dir; //类似打开文件的fd描述符
struct dirent *pdr; //man readdir可以看到此结构
if(lstat(w,&buf)<) //获取文件/目录属性并赋值给buf,该函数和lstat一样,只是当w为链接时,指代他本身,并不存在文件
{
fprintf(stderr,"stat error:%s\n",strerror(errno));
return -;
}
if(S_ISDIR(buf.st_mode)) //判断是否为目录,是返回真
{
dir = opendir(w); //打开目录
while ((pdr = readdir(dir))!=NULL) //读/遍历目录
{
if(aflag==) //如果不带a参数,越过以.开头的所有文件/目录
{
if(pdr->d_name[]=='.')
continue;
memset(name,,);
strcpy(name,w); //拷贝
strcat(name,"/"); //追加
strcat(name,pdr->d_name);
do_ls(name,pdr->d_name,lflag);
}
else //有a参数显示所有
{
memset(name,,);
strcpy(name,w);
strcat(name,"/");
strcat(name,pdr->d_name);
do_ls(name,pdr->d_name,lflag);
}
}
closedir(dir);
}
else //为文件则直接显示
{
do_ls(w,w,lflag);
}
return ;
}
int main(int argc,char **argv)
{
int aflag =;
int lflag =;
char c;
int i;
while((c = getopt(argc,argv,"al"))!=-) //解析命令行参数,即-/--后面的字符串和给定的字符串匹配,有未解析字母返回字母或问号(取决于第3个参数),否则返回-1
{
switch(c) //此处仅匹配a(所有)和l(列表),即只支持参数a、l
{
case 'a':
aflag =;
break;
case 'l':
lflag =;
break;
default:
break;
}
}
if(argc == optind )
{
ls_prepare("./",aflag,lflag);
}
else
{
for(i=optind; i<argc; i++) //所有目录都传进去
ls_prepare(argv[i],aflag,lflag);
}
printf("\n");
return ;
}
ls -l功能:
#include <stdio.h>
#include <sys/types.h>
#include <dirent.h>
#include <sys/stat.h>
#include <pwd.h>
#include <grp.h> 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_to_letters(info_p->st_mode, modestr); printf("%s", modestr);
printf(" %4d", (int) info_p->st_nlink);
printf(" %-8s", uid_to_name(info_p->st_uid));
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); } 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';
}
} 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 numstr;
}else{
return pw_ptr->pw_name;
}
} 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;
}
}
void do_ls(char dirname[]) {
DIR* dir_ptr;
struct dirent* direntp; if ((dir_ptr = opendir(dirname)) == NULL) {
fprintf(stderr, "ls2: cannot open %s \n", dirname);
} else {
while ((direntp = readdir(dir_ptr)) != NULL) {
dostat(direntp->d_name);
} close(dir_ptr);
}
} void dostat(char* filename) {
struct stat info; if (stat(filename, &info) == -1) {
perror(filename);
} else {
show_file_info(filename, &info);
}
} int main(int ac,char* av[]){
if(ac == 1){
do_ls(".");
}else{
while(--ac){
printf("%s: \n",*++av);
do_ls(*av);
}
}
}
用C++实现Linux中shell的ls功能的更多相关文章
- Linux中Shell
Linux中Shell Shell是什么 Shell是一个命令行解释器,为用户提供了一个向Linux内核发送请求以便运行程序的界面系统级程序,可以用Shell来启动.挂起.停止.编写一些程序. S ...
- linux中shell变量$#,$@,$0,$1,$2的含义解释
linux中shell变量$#,$@,$0,$1,$2的含义解释: 变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行 ...
- linux中shell变量$#,$@,$0,$1,$2的含义解释
linux中shell变量$#,$@,$0,$1,$2的含义解释 linux中shell变量$#,$@,$0,$1,$2的含义解释: 变量说明: $$ Shell本身的PID(ProcessID ...
- Linux中shell变量$0,$?等含义
linux中shell变量$#,$@,$0,$1,$2的基本含义: 变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行 ...
- 【转】linux中shell变量$#,$@,$0,$1,$2的含义解释
原文网址:http://www.cnblogs.com/fhefh/archive/2011/04/15/2017613.html linux中shell变量$#,$@,$0,$1,$2的含义解释: ...
- 【Shell】linux中shell变量$#,$@,$0,$1,$2的含义解释 && set 关键字使用
linux中shell变量$#,$@,$0,$1,$2的含义解释 摘抄自:ABS_GUIDE 下载地址:http://www.tldp.org/LDP/abs/abs-guide.pdf linu ...
- linux中shell变量$#,$@,$0,$1,$2的含义
linux中shell变量$#,$@,$0,$1,$2的含义解释: 变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行 ...
- linux中shell变量$#,$@,$0,$1,$2
linux中shell变量$#,$@,$0,$1,$2的含义解释: 变量说明: $$ Shell本身的PID(ProcessID) $! Shell最后运行的后台Process的PID $? 最后运行 ...
- 浅谈linux中shell变量$#,$@,$0,$1,$2,$?的含义解释
浅谈linux中shell变量$#,$@,$0,$1,$2,$?的含义解释 下面小编就为大家带来一篇浅谈linux中shell变量$#,$@,$0,$1,$2的含义解释.小编觉得挺不错的,现在就分享给 ...
随机推荐
- In-Memory:在内存中创建临时表和表变量
在Disk-Base数据库中,由于临时表和表变量的数据存储在tempdb中,如果系统频繁地创建和更新临时表和表变量,大量的IO操作集中在tempdb中,tempdb很可能成为系统性能的瓶颈.在SQL ...
- Entity Framework教程(第二版)
源起 很多年前刚毕业那阵写过一篇关于Entity Framework的文章,没发首页却得到100+的推荐.可能是当时Entity Framework刚刚发布介绍EF的文章比较少.一晃这么多年过去了,E ...
- Socket聊天程序——客户端
写在前面: 上周末抽点时间把自己写的一个简单Socket聊天程序的初始设计和服务端细化设计记录了一下,周二终于等来毕业前考的软考证书,然后接下来就是在加班的日子度过了,今天正好周五,打算把客户端的详细 ...
- pt-table-checksum
pt-table-checksum是percona公司提供的一个用于在线比对主从数据一致性的工具. 实现原理 将一张大表分成多个chunk,每次针对一个chunk进行校验,同时将校验的结果通过REPL ...
- SQL Server 无法连接到服务器。SQL Server 复制需要有实际的服务器名称才能连接到服务器。请指定实际的服务器名称。
异常处理汇总-数据库系列 http://www.cnblogs.com/dunitian/p/4522990.html SQL性能优化汇总篇:http://www.cnblogs.com/dunit ...
- js:给定两个数组,如何判断他们的相对应下标的元素类型是一样的
题目: 给Array对象原型上添加一个sameStructureAs方法,该方法接收一个任意类型的参数,要求返回当前数组与传入参数数组(假定是)相对应下标的元素类型是否一致. 假设已经写好了Array ...
- ASP.NET Core CORS 简单使用
CORS 全称"跨域资源共享"(Cross-origin resource sharing). 跨域就是不同域之间进行数据访问,比如 a.sample.com 访问 b.sampl ...
- 当web.config文件放置在共享目录下(UNC),启动IIS会提示有错误信息500.19,伴随有错误代码0x80070003和错误代码0x80070005的解决办法
最近遇到一个很有意思的使用环境,操作人员将所有的网站应用内容投放到共享存储里面,并且使用微软的SMB协议将其以CIFS的方式共享出来,使用Windows Server 2008 R2的IIS将其连接起 ...
- 搭建属于自己的VIP积分系统(1)
很久没写博客了,如果有写得不好的地方,还请多多见谅. 架构设计 需求分析 这篇文章主要是介绍此VIP系统的基础架构.说实在的,我其实对 架构方面也不是很懂,我这套框架 还是拿别人的东西改过来的,并不是 ...
- .NET应用程序域
在.NET平台下,可执行程序并没有直接承载在Windows进程中,而非托管程序是直接承载的..NET可执行程序承载在进程的一个逻辑分区中,称之为应用程序域(AppDomain).一个进程可以包含多个应 ...