文件I/O实践(2) --文件stat
功能:获取文件元数据
#include <sys/types.h> #include <sys/stat.h> #include <unistd.h> int stat(const char *path, struct stat *buf); int fstat(int fd, struct stat *buf); int lstat(const char *path, struct stat *buf);
stat结构体
struct stat { dev_t st_dev; /* ID of device containing file */ ino_t st_ino; /* inode number */ mode_t st_mode; /* protection */ nlink_t st_nlink; /* number of hard links */ uid_t st_uid; /* user ID of owner */ gid_t st_gid; /* group ID of owner */ dev_t st_rdev; /* device ID (if special file) */ off_t st_size; /* total size, in bytes */ blksize_t st_blksize; /* blocksize for filesystem I/O */ blkcnt_t st_blocks; /* number of 512B blocks allocated */ time_t st_atime; /* time of last access */ time_t st_mtime; /* time of last modification */ time_t st_ctime; /* time of last status change */ };
//示例 - err_exit函数如前 // 其实可以通过Linux 系统调用major, minor来替换(如下) #define MAJOR(a) (int)((unsigned short)a >> 8) //主设备号: 获取高8位 #define MINOR(a) (int)((unsigned short)a & 0xFF)//次设备号: 获取低8位 bool fileType(const struct stat &fileStat); void filePerm(const struct stat &fileStat, char *perm); int main(int argc, char *argv[]) { if (argc != 2) { cerr << "Usage: " << argv[0] << " <file-name>" << endl; exit(EXIT_FAILURE); } struct stat fileStat; if (lstat(argv[1], &fileStat) == -1) err_exit("stat error"); cout << "file-name: " << argv[1] << endl; cout << "st_ino = " << fileStat.st_ino << endl; cout << "device major: " << major(fileStat.st_dev) << ", minor: " << minor(fileStat.st_dev) << endl; if (fileType(fileStat)) { cout << "----------------------------" << endl; cout << "major: " << MAJOR(fileStat.st_dev) << ", minor: " << MINOR(fileStat.st_rdev) << endl; // cout << "major: " << major(fileStat.st_dev) // << ", minor: " << minor(fileStat.st_rdev) << endl; } //获取文件的权限: 但要注意需要&上07777 fprintf(stdout, "file permission: %o", fileStat.st_mode&07777); char perm[11]; filePerm(fileStat, perm); cout << ", msg: " << perm << endl; cout << "st_nlink = " << fileStat.st_nlink << endl; cout << "st_uid = " << fileStat.st_uid << endl; cout << "st_gid = " << fileStat.st_gid << endl; cout << "st_size = " << fileStat.st_size << endl; cout << "st_blksize = " << fileStat.st_blksize << endl; cout << "st_blocks = " << fileStat.st_blocks << endl; cout << "st_atime = " << fileStat.st_atime << endl; cout << "st_ctime = " << fileStat.st_ctime << endl; cout << "st_mtime = " << fileStat.st_mtime << endl; } bool fileType(const struct stat &fileStat) { cout << "file-type: "; switch(fileStat.st_mode & S_IFMT) { case S_IFSOCK: cout << "socket"; break; case S_IFLNK: cout << "symbolic link"; break; case S_IFREG: cout << "regular file"; break; case S_IFBLK: cout << "block device" << endl; return true; break; case S_IFDIR: cout << "directory"; break; case S_IFCHR: cout << "character device" << endl; return true; break; case S_IFIFO: cout << "FIFO" << endl; break; default: cout << "unknown?"; break; } cout << endl; return false; } void filePerm(const struct stat &fileStat, char *perm) { strcpy(perm, "----------"); switch(fileStat.st_mode & S_IFMT) { case S_IFSOCK: perm[0] = 's'; break; case S_IFLNK: perm[0] = 'l'; break; case S_IFREG: perm[0] = '-'; break; case S_IFBLK: perm[0] = 'b'; break; case S_IFDIR: perm[0] = 'd'; break; case S_IFCHR: perm[0] = 'c'; break; case S_IFIFO: perm[0] = 'p'; break; default: perm[0] = '?'; break; } if (fileStat.st_mode & S_IRUSR) perm[1] = 'r'; if (fileStat.st_mode & S_IWUSR) perm[2] = 'w'; if (fileStat.st_mode & S_IXUSR) perm[3] = 'x'; if (fileStat.st_mode & S_IRGRP) perm[4] = 'r'; if (fileStat.st_mode & S_IWGRP) perm[5] = 'w'; if (fileStat.st_mode & S_IXGRP) perm[6] = 'x'; if (fileStat.st_mode & S_IROTH) perm[7] = 'r'; if (fileStat.st_mode & S_IWOTH) perm[8] = 'w'; if (fileStat.st_mode & S_IXOTH) perm[9] = 'x'; }
[拓展]
1.getpwuid
struct passwd *getpwuid(uid_t uid);
//passwd结构体 struct passwd { char *pw_name; /* username */ char *pw_passwd; /* user password */ uid_t pw_uid; /* user ID */ gid_t pw_gid; /* group ID */ char *pw_gecos; /* user information */ char *pw_dir; /* home directory */ char *pw_shell; /* shell program */ };
2.getgrgid
struct group *getgrgid(gid_t gid);
//group结构体 struct group { char *gr_name; /* group name */ char *gr_passwd; /* group password */ gid_t gr_gid; /* group ID */ char **gr_mem; /* group members */ };
3. readlink
ssize_t readlink(const char *path, char *buf, size_t bufsiz);
4. localtime
struct tm *localtime(const time_t *timep);
//tm结构体 struct tm { int tm_sec; /* seconds */ int tm_min; /* minutes */ int tm_hour; /* hours */ int tm_mday; /* day of the month */ int tm_mon; /* month */ int tm_year; /* year */ int tm_wday; /* day of the week */ int tm_yday; /* day in the year */ int tm_isdst; /* daylight saving time */ };
//示例: 实现简单的ls -l功能 #include <iostream> #include <stdio.h> #include <stdlib.h> #include <string.h> #include <string.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> #include <unistd.h> #include <grp.h> #include <pwd.h> #include <time.h> using namespace std; inline void err_exit(std::string message); bool isDevice(const struct stat &fileStat); bool isLink(const struct stat &fileStat); void filePerm(const struct stat &fileStat, char *perm); int main(int argc, char *argv[]) { if (argc != 2) { cerr << "Usage: " << argv[0] << " <file-name>" << endl; exit(EXIT_FAILURE); } struct stat fileStat; if (lstat(argv[1], &fileStat) == -1) err_exit("lstat error"); //获取权限 char perm[11]; filePerm(fileStat, perm); cout << perm << ' '; //获取文件链接数 cout << fileStat.st_nlink << ' '; //获取文件所有者 struct passwd *ps = getpwuid(fileStat.st_uid); cout << ps->pw_name << ' '; //获取文件所属组 struct group *gp = getgrgid(fileStat.st_gid); cout << gp->gr_name << ' '; if (isDevice(fileStat)) cout << major(fileStat.st_dev) << ", " << minor(fileStat.st_rdev) << ' '; else cout << fileStat.st_size << ' '; // 打印最后一次修改时间 time_t mtime = fileStat.st_mtime; struct tm *pTime = localtime(&mtime); cout << pTime->tm_mon+1 << "月 " << pTime->tm_mday << ' ' << pTime->tm_hour << ':' << pTime->tm_min << ' '; // 打印文件名 cout << argv[1]; if (isLink(fileStat)) { cout << " -> "; char name[1024] = {0}; if (readlink(argv[1], name, sizeof(name)) == -1) err_exit("readlink error"); cout << name; } cout << endl; } inline void err_exit(std::string message) { perror(message.c_str()); exit(EXIT_FAILURE); }
bool isDevice(const struct stat &fileStat) { switch(fileStat.st_mode & S_IFMT) { case S_IFBLK: case S_IFCHR: return true; break; default: return false; break; } return false; }
bool isLink(const struct stat &fileStat) { if ((fileStat.st_mode & S_IFMT) == S_IFLNK) return true; return false; }
void filePerm(const struct stat &fileStat, char *perm) { strcpy(perm, "----------"); switch(fileStat.st_mode & S_IFMT) { case S_IFSOCK: perm[0] = 's'; break; case S_IFLNK: perm[0] = 'l'; break; case S_IFREG: perm[0] = '-'; break; case S_IFBLK: perm[0] = 'b'; break; case S_IFDIR: perm[0] = 'd'; break; case S_IFCHR: perm[0] = 'c'; break; case S_IFIFO: perm[0] = 'p'; break; default: perm[0] = '?'; break; } if (fileStat.st_mode & S_IRUSR) perm[1] = 'r'; if (fileStat.st_mode & S_IWUSR) perm[2] = 'w'; if (fileStat.st_mode & S_IXUSR) perm[3] = 'x'; if (fileStat.st_mode & S_IRGRP) perm[4] = 'r'; if (fileStat.st_mode & S_IWGRP) perm[5] = 'w'; if (fileStat.st_mode & S_IXGRP) perm[6] = 'x'; if (fileStat.st_mode & S_IROTH) perm[7] = 'r'; if (fileStat.st_mode & S_IWOTH) perm[8] = 'w'; if (fileStat.st_mode & S_IXOTH) perm[9] = 'x'; }