1.指定路径下的所有文件属性打印出来,类似 ls -l 一个指定的目录
//代码
#include <stdio.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <time.h>
#include <errno.h>
#include <dirent.h>
#include <pwd.h>
#include <grp.h>
/*--------------文件权限------------------*/
void get_FilePermission(mode_t mode)
{
for(int i=0,t=0400;i<9;i++,t=t>>1)
{
if((mode&t) && i%3==0)
putchar('r');
else if((mode&t) && i%3==1)
putchar('w');
else if((mode&t) && i%3==2)
putchar('x');
else
putchar('-');
}
}
/*----------------文件类型--------------------*/
void get_filetype(mode_t m)
{
#if 0
if(S_ISREG(m))
putchar('-');
else if(S_ISDIR(m))
putchar('d');
else if(S_ISCHR(m))
putchar('c');
else if(S_ISBLK(m))
putchar('b');
else if(S_ISFIFO(m))
putchar('p');
else if(S_ISLNK(m))
putchar('l');
else if(S_ISSOCK(m))
putchar('s');
#else
switch(m & S_IFMT)
{
case S_IFSOCK: putchar('s');break;
case S_IFLNK: putchar('l');break;
case S_IFREG: putchar('-');break;
case S_IFBLK: putchar('b');break;
case S_IFDIR: putchar('d');break;
case S_IFCHR: putchar('c');break;
case S_IFIFO: putchar('p');break;
}
#endif
}
/*-----------------用户名-------------------*/
void get_usrname(uid_t uid)
{
struct passwd *pwd = getpwuid(uid);
if(NULL == pwd)
{
perror("getpwuid");
return;
}
printf(" %s",pwd->pw_name);
}
/*-----------------用户组名------------------*/
void get_grpname(gid_t gid)
{
struct group *grp = getgrgid(gid);
if(NULL == grp)
{
perror("getgrgid");
return;
}
printf(" %s",grp->gr_name);
}
void FileInfo(char *pathname,char *filename)
{
struct stat buf;
if(stat(pathname,&buf) < 0)
{
perror("stat");
return;
}
//文件类型和权限
get_filetype(buf.st_mode);
get_FilePermission(buf.st_mode);
//硬链接数
printf(" %ld",buf.st_nlink);
//用户和用户组
get_usrname(buf.st_uid);
get_grpname(buf.st_gid);
//文件大小
printf(" %5ld",buf.st_size);
//时间
struct tm time;
localtime_r(&buf.st_mtime,&time);
printf(" %d.%02d.%02d %02d:%02d:%02d",time.tm_year+1900,time.tm_mon+1,time.tm_mday,time.tm_hour,time.tm_min,time.tm_sec);
//文件名
printf(" %s\n", filename);
}
int main(int argc, const char *argv[])
{
if(argc < 2)
{
printf("请输入路径\n");
return -1;
}
DIR *dp = opendir(argv[1]);
if(NULL == dp)
{
perror("opendir");
return -1;
}
while(1)
{
struct dirent *rp = readdir(dp);
if(NULL == rp)
{
if(0 == errno)
break;
else
return -1;
}
if(rp->d_name[0] == '.')
continue;
//文件路径拼接,补'/'
char path[256] = "";
strcpy(path,argv[1]);
int i = strlen(path);
if(path[i-1]!='/')
path[i] = '/';
strcat(path,rp->d_name);
FileInfo(path,rp->d_name);
}
closedir(dp);
return 0;
}
运行结果
2.验证
任务1:fork前创建一个int a,父子进程中是否都有变量a,虚拟地址是否相同,物理地址是否相同
任务2:fork函数后,在父进程中int b,父子进程中是否都有变量b,虚拟地址是否相同,物理地址是否相同
任务3:fork函数后,在子进程中int c,父子进程中是否都有变量c,虚拟地址是否相同,物理地址是否相同
(1)父子进程中都存在变量a,并且虚拟地址相同,但物理地址不相同
//代码
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <sys/types.h>
int main(int argc, const char *argv[])
{
int a = 10;
int *p = NULL;
pid_t cpid = fork();
printf("cpid=%d __%d__\n",cpid, __LINE__);
if(cpid > 0)
{
a = 20;
printf("父进程: a=%d &a=%p\n",a,&a);
}
else if(cpid == 0)
{
printf("子进程: a=%d &a=%p\n",a,&a);
}
else
{
perror("fork");
return -1;
}
return 0;
}
//运行结果
zzy@zzy-vm:~/hqyj/IO/day4$ ./a.out
cpid=3968 __12__
父进程: a=20 &a=0x7ffe727277c8
cpid=0 __12__
子进程: a=10 &a=0x7ffe727277c8
从运行结果可以看出,父子进程中都存在变量a并且虚拟地址相同,但是在父进程中修改变量a后并不会改变子进程中变量a的值,所以两者的物理地址不同。
(2)仅在父进程中存在变量b
(3)仅在子进程中存在变量c