【本文谢绝转载,原文来自http://990487026.blog.51cto.com】
Linux系统开发 文件操作 ext2文件系统了解 stat()函数 access()函数 chmod()函数 utime()函数 truncate()函数 link()硬链接函数 symlink()软链接函数 readlink()函数 unlink函数 rename函数() 目录操作 chdir()/fchdir() getcwd/getwd()/get_current_dir_name() pathconf() opendir() readdir() dup()/dup2() 练习:
stat函数获取文件大小
chunli@ubuntu16:~/linux_c$ cat main.c
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
//#include <fcntl.h>
int main(int argc,char **argv)
{
struct stat s_buf;
if(argc < 2)
{
printf("参数不够\n");
exit(1);
}
if(stat(argv[1],&s_buf) < 0)
{
perror("stat");
exit(2);
}
printf("文件名:%s\n",argv[1]);
printf("文件大小:%ld字节 \n",s_buf.st_size);
}
// int stat(const char *pathname, struct stat *buf);
// int fstat(int fd, struct stat *buf);
// int lstat(const char *pathname, struct stat *buf);
// lstat不跟踪符号链接
//
// 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 */
//
// /* Since Linux 2.6, the kernel supports nanosecond
// precision for the following timestamp fields.
// For the details before Linux 2.6, see NOTES. */
//
// struct timespec st_atim; /* time of last access */
// struct timespec st_mtim; /* time of last modification */
// struct timespec st_ctim; /* time of last status change */
//
// #define st_atime st_atim.tv_sec /* Backward compatibility */
// #define st_mtime st_mtim.tv_sec
// #define st_ctime st_ctim.tv_sec
// };
//
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out main.c
文件名:main.c
文件大小:1454字节
chunli@ubuntu16:~/linux_c$access()文件访问函数,测试文件是否存在程序
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out main.c
./main.c 存在
chunli@ubuntu16:~/linux_c$
chunli@ubuntu16:~/linux_c$
chunli@ubuntu16:~/linux_c$ cat main.c
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
//#include <fcntl.h>
int main(void)
{
if(access("./main.c",F_OK) < 0)
{
perror("open ./haha.txt");
exit(1);
}
printf("./main.c 存在\n");
}
// int access(const char *pathname, int mode);
// The mode specifies the accessibility check(s) to be performed,
// and is either the value F_OK, or a mask consisting of the bitwise
// OR of one or more of R_OK, W_OK,and X_OK. F_OK tests for the
// existence of the file. R_OK, W_OK, and X_OK test whether the
// file exists and grants read, write, and execute permissions, respec‐tively.
//RETURN VALUE:
// On success (all requested permissions granted,
// or mode is F_OK and the file exists), zero is returned.
// On error (at least one bit in mode asked for a permission ,
// that is denied, or mode is F_OK and the file does not exist,
// or some other error occurred), -1 is returned, and errno is set appropriately.
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out main.c
./main.c 存在
chunli@ubuntu16:~/linux_c$黏住位:用户以root权限执行
chunli@ubuntu16:~$ ll /usr/bin/passwd -rwsr-xr-x 1 root root 53K 3月 29 17:25 /usr/bin/passwd*
chmod()函数
chunli@ubuntu16:~/linux_c$ cat main.c
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
//#include <fcntl.h>
int main(int argc,char **argv)
{
if(argc < 3)
{
printf("参数不够!\n");
}
//需要测试文件是否存在
//"0222" => 转8进制
int mode = atoi(argv[2]);
chmod(argv[1],mode);
}
//需要完善的地方:
//1 字符串转8进制
//2 判断文件是否存在才应该去chmod
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out file 111
chunli@ubuntu16:~/linux_c$ ll
总用量 16K
-rwxrwxr-x 1 chunli chunli 8.5K 8月 4 09:59 a.out*
---xr-xrwx 1 chunli chunli 0 8月 4 09:55 file*
-rw-rw-r-- 1 chunli chunli 435 8月 4 09:59 main.c
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out file 511
chunli@ubuntu16:~/linux_c$ ll
总用量 16K
-rwxrwxr-x 1 chunli chunli 8.5K 8月 4 10:00 a.out*
-rwxrwxrwx 1 chunli chunli 0 8月 4 09:55 file*
-rw-rw-r-- 1 chunli chunli 435 8月 4 09:59 main.c
chunli@ubuntu16:~/linux_c$utime()更新文件时间函数
// #include <sys/types.h> // #include <utime.h> // // int utime(const char *filename, const struct utimbuf *times); // // #include <sys/time.h> // // int utimes(const char *filename, const struct timeval times[2]);
truncate()函数,截断文件
// #include <unistd.h> // #include <sys/types.h> // // int truncate(const char *path, off_t length); // int ftruncate(int fd, off_t length);
2.7.1 link
创建一个硬链接
当rm删除文件时,只是删除了目录下的记录项和把inode硬链接计数减1,当硬链接计数
减为0时,才会真正的删除文件。
#include <unistd.h>
int link(const char *oldpath, const char *newpath);
* 硬链接通常要求位于同一文件系统中,POSIX允许夸文件系统
* 符号链接没有文件系统限制
* 通常不允许创建目录的硬链接,某些unix系统下超级用户可以创建目录的硬链
* 创建目录项以及增加硬链接计数应当是一个原子操作
2.7.2 symlink
int symlink(const char *oldpath, const char *newpath)
2.7.3 readlink
读符号链接所指向的文件名字,不读文件内容
ssize_t readlink(const char *path, char *buf, size_t bufsiz)
2.7.4 unlink
int unlink(const char *pathname)
1. 如果是符号链接,删除符号链接
2. 如果是硬链接,硬链接数减1,当减为0时,释放数据块和inode
3. 如果文件硬链接数为0,但有进程已打开该文件,并持有文件描述符,则等该进程关闭该文件时,kernel才真正
去删除该文件
4. 利用该特性创建临时文件,先open或creat创建一个文件,马上unlink此文件
rm 底层也是调用unlink函数
2.8 rename
文件重命名
#include <stdio.h>
int rename(const char *oldpath, const char *newpath);
2.9 chdir
#include <unistd.h>
int chdir(const char *path);
int fchdir(int fd);
改变当前进程的工作目录
2.10 getcwd
获取当前进程的工作目录
#include <unistd.h>
char *getcwd(char *buf, size_t size);
chdir(),getcwd() char *getcwd(char *buf, size_t size);//用户自己定义字符数组 char *getwd(char *buf); //没有数组大小检查 char *get_current_dir_name(void);//函数静态数组或者堆空间
获取文件路径
chunli@ubuntu16:~/linux_c$ cat main.c
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
int main(int argc,char **argv)
{
char buf [10] = {0};
chdir("/home");
getcwd(buf,sizeof(buf));
printf("%s \n",buf);
}
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out file 511
/home
chunli@ubuntu16:~/linux_c$pathconf 文件配置信息
#include <unistd.h>
long fpathconf(int fd, int name);
long pathconf(char *path, int name);
chunli@ubuntu16:~/linux_c$ cat main.c
#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <unistd.h>
#include <sys/stat.h>
#include <sys/types.h>
int main(int argc,char **argv)
{
printf("%ld\n",fpathconf(STDOUT_FILENO,_PC_NAME_MAX));
printf("%ld\n",pathconf("./main.c",_PC_NAME_MAX));
return 0;
}
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out file 511
255
255
chunli@ubuntu16:~/linux_c$opendir() readdir() 目录遍历
struct dirent
struct stat
chunli@ubuntu16:~/linux_c$ cat main.c
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
#include <dirent.h>
#include <stdio.h>
#include <string.h>
#define MAX_PATH 1024
/* dirwalk: apply fcn to all files in dir */
void dirwalk(char *dir, void (*fcn)(char *)) //遍历目录
{
char name[MAX_PATH];
struct dirent *dp;
DIR *dfd;
if ((dfd = opendir(dir)) == NULL) //如果打开目录失败
{
fprintf(stderr, "dirwalk: can‘t open %s\n", dir);
return;
}
while ((dp = readdir(dfd)) != NULL) //读取目录内容
{
if (strcmp(dp->d_name, ".") == 0 || strcmp(dp->d_name, "..") == 0)
{
continue; /* skip self and parent */
}
if (strlen(dir)+strlen(dp->d_name)+2 > sizeof(name))// 杠与0"/home/chunli\0"
{
fprintf(stderr, "dirwalk: name %s %s too long\n",dir, dp->d_name);
}
else
{
sprintf(name, "%s/%s", dir, dp->d_name);
(*fcn)(name);
}
}
closedir(dfd);
}
/* fsize: print the size and name of file "name" */
void fsize(char *name) //显示文件的大小
{
struct stat stbuf;
if (stat(name, &stbuf) == -1)
{
fprintf(stderr, "fsize: can‘t access %s\n", name);
return;
}
if ((stbuf.st_mode & S_IFMT) == S_IFDIR) //判断是不是目录
{
dirwalk(name, fsize); //如果是目录就递归
}
printf("%8ld字节 %s\n", stbuf.st_size, name);
}
int main(int argc, char **argv)
{
if (argc == 1) /* default: current directory */
{
fsize(".");
}
else
{
while (--argc > 0) //这样做,可以把程序后面的参数全部遍历
{
fsize(*(++argv));
}
}
return 0;
}
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out ../linux_c/
0字节 ../linux_c//file
1504字节 ../linux_c//main.c
12288字节 ../linux_c//.main.c.swp
9232字节 ../linux_c//a.out
4096字节 ../linux_c/
chunli@ubuntu16:~/linux_c$dup/dup2
#include <unistd.h>
int dup(int oldfd);
int dup2(int oldfd, int newfd);
dup和dup2都可用来复制一个现存的文件描述符,使两个文件描述符指向同一个file结
构体。如果两个文件描述符指向同一个file结构体,File Status Flag和读写位置只保存一
份在file结构体中,并且file结构体的引用计数是2。如果两次open同一文件得到两个文件
描述符,则每个描述符对应一个不同的file结构体,可以有不同的File Status Flag和读写
位置。请注意区分这两种情况。
chunli@ubuntu16:~/linux_c$ cat main.c
#include <unistd.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
int main(void)
{
int fd, save_fd;
char msg[] = "This is a test\n";
fd = open("somefile", O_RDWR|O_CREAT, S_IRUSR|S_IWUSR);
if(fd<0) {
perror("open");
exit(1);
}
printf("fd = %d\n",fd);
save_fd = dup(STDOUT_FILENO);
printf("save_fd = %d\n",save_fd);
dup2(fd, STDOUT_FILENO); //这个步骤会关闭STDOUT_FILENO, fd就是标准输出了
printf("第3次 fd = %d\n",fd);
close(fd);
write(STDOUT_FILENO, msg, strlen(msg));
dup2(save_fd, STDOUT_FILENO);
write(STDOUT_FILENO, msg, strlen(msg));
close(save_fd);
return 0;
}
chunli@ubuntu16:~/linux_c$ gcc main.c && ./a.out
fd = 3
save_fd = 4
This is a test
chunli@ubuntu16:~/linux_c$ cat somefile
第3次 fd = 3
This is a test
chunli@ubuntu16:~/linux_c$2.14 练习
1. 实现ls -l功能,可以解析文件权限位。
2. 实现ls -l功能,可以解析出文件所有者和文件所有组。(偏难)
3. 实现rm删除命令,如
rm file
rm directory
*注意,千万不要在你有代码的目录下做测试,防止删除你的有用文件,友情提示(rmdir/unlink和递归遍历目
录)
4. 从文件里面读出1000个随机数,进行排序,再写到另一文件中。(考虑使用重定向dup/dup2)
本文出自 “魂斗罗” 博客,谢绝转载!
原文地址:http://990487026.blog.51cto.com/10133282/1834411