标准C库IO函数
FILE *fopen(const char *filename, const char *mode)
- 返回值(FILE * fp):文件描述符(文件指针),索引到对应的磁盘文件
- 文件读写指针位置:在文件内部有一个位置指针,用来指向当前读写到的位置,也就是读写到第几个字节。在文件打开时,该指针总是指向文件的第一个字节。
注意:这个文件内部的位置指针与C语言中的指针不是一回事。位置指针仅仅是一个标志,表示文件读写到的位置,也就是读写到第几个字节,它不表示地址。文件每读写一次,位置指针就会移动一次,它不需要你在程序中定义和赋值,而是由系统自动设置,对用户是隐藏的。
- I/O缓冲区(内存地址):通过寻址找到对应的内存块并将数据从内存刷新到磁盘
刷新方法:
- 刷新缓冲区:fflush
- 缓冲区已满(默认Buffer 8192byte)
- 正常关闭文件:a.fclose b.return(main函数) c.exit(main函数)
- 直接调用标准C库的I/O会有缓冲区存在,在系统内部传递文件时效率高(可以减少系统I/O执行次数)
- 在网络传输时选用系统I/O可以提高网络通信效率(及时发送文件)
文件打开、创建、关闭
- 头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
- 函数
int open(const char *pathname, int flags);
- 作用:打开一个已经存在的文件
- 参数:
- pathname:要打开的文件路径(必选项)
- flags:对文件的操作权限设置还有其他的设置
- 必选项:O_RDONLY, O_WRONLY, O_RDWR 这三个权限是互斥的
- 可选项:O_CREAT 文件不存在,创建新文件
- 返回值:返回一个新的文件描述符,如果调用失败,返回-1
- 函数
int open(const char *pathname, int flags, mode_t mode);
- 参数:
- mode:八进制的数,表示创建出的新的文件的操作权限,比如:0775
-
最终的权限:mode & ~umask(umask:抹去某些权限)
-
函数
close(fd);
- 作用:关闭文件
- 函数
#include <stdio.h>
void perror(const char *s);
- 作用:打印errno对应的错误描述
- errno:属于Linux系统函数库,库里面是一个全局变量,记录的是最近的错误号
- 参数:用户描述,比如hello,最终输出的内容是hello:xxx(实际的错误描述)
int fd = open("a.txt", O_RDWR | O_CREAT, 0777);
if(fd == -1){
perror("open");
}
作用:创建并打开一个文件,并赋给其相应的权限,如果失败就返回对应的错误描述
文件读取、写入操作
- 头文件
#include <unistd.h>
- 函数
ssize_t read(int fd, void *buf, size_t count);
- 参数:
-
buf:缓冲区,需要读取数据存放的地方,数组的地址(传出参数)
-
注:传入参数为本身有值,传入函数让函数使用,传出参数是本身没值,传入函数让函数赋值。
-
-
count:指定的数组的大小
-
- 返回值:
- 成功:
-
>0:返回实际的读取到的字节数
-
-0:文件已经读取完了
-
- 成功:
-
函数:
ssize_t write(int fd, const void *buf, size_t count);
- 返回值:
- 成功:实际写入的字节数
- 失败:返回-1,并设置errno
文件指针移动函数
- 头文件:
#include <sys/types.h>
#include <unistd.h>
- 函数:
off_t lseek(int fd, off_t offset, int whence);
- 参数:
-
offset:偏移量
-
whence:
-
SEEK_SET:设置文件指针的偏移量
-
SEEK_CUR:设置偏移量:当前位置 + 第二个参数offset的值
-
SEEK_END:设置偏移量:文件大小 + 第二个参数offset的值
-
-
- 返回值:返回文件指针的位置,错误返回-1
- 作用:
-
移动文件指针到头文件:lseek(fd, 0, SEEEK_SET);
-
获取当前文件指针的位置:lseek(fd, 0, SEEK_CUR);
-
获取文件长度:lseek(fd, 0, SEEK_END);
-
拓展文件的长度,当前文件10b,110b增加了100个字节:lseek(fd, 100, SEEK_END);
-
查看文件信息函数
- 头文件:
#include <sys/types.h>
#include <sys/stat.h>
#include <unistd.h>
- 函数:
int stat(const char *pathname, struct stat *statbuf);
- 作用:获取一个文件相关的信息
- 参数:
-
pathname:文件路径
-
statbuf:结构体变量,传出参数,用于保存获取到的文件信息
-
- 返回值:
-
成功:返回0
-
失败:返回-1,设置errno
-
- 函数:
int lstat(const char *pathname, struct stat *statbuf);
- 作用:获取软连接文件的信息(如果用stat输出软连接文件的信息会输出所指向的文件的信息)
模拟实现ls -l命令
#include <sys/stat.h>
#include <sys/types.h>
#include <unistd.h>
#include <stdio.h>
#include <pwd.h>
#include <grp.h>
#include <time.h>
#include <string.h>
int main(int argc, char *argv[]){
//判断输入的参数是否正确
if(argc < 2){
printf("%s filename\n", argv[0]);
return -1;
}
//通过stat函数获取用户传入的文件的信息
struct stat st;
int ret = stat(argv[1], &st);
if(ret == -1){
perror("stat");
return -1;
}
//获取文件类型和文件权限
char perms[11] = {0};//用于保存文件类型和文件权限的字符串
switch(st.st_mode & S_IFMT){
case S_IFLNK:
perms[0] = 'l';
break;
case S_IFDIR:
perms[0] = 'd';
break;
case S_IFREG:
perms[0] = '-';
break;
case S_IFBLK:
perms[0] = 'b';
break;
case S_IFCHR:
perms[0] = 'c';
break;
case S_IFSOCK:
perms[0] = 's';
break;
case S_IFIFO:
perms[0] = 'p';
break;
default:
perms[0] = '?';
break;
}
//判断文件的访问权限
//文件所有者
perms[1] = (st.st_mode & S_IRUSR) ? 'r' : '-';
perms[2] = (st.st_mode & S_IWUSR) ? 'w' : '-';
perms[3] = (st.st_mode & S_IXUSR) ? 'x' : '-';
//文件所在组
perms[4] = (st.st_mode & S_IRGRP) ? 'r' : '-';
perms[5] = (st.st_mode & S_IWGRP) ? 'w' : '-';
perms[6] = (st.st_mode & S_IXGRP) ? 'x' : '-';
//其他人
perms[7] = (st.st_mode & S_IROTH) ? 'r' : '-';
perms[8] = (st.st_mode & S_IWOTH) ? 'w' : '-';
perms[9] = (st.st_mode & S_IXOTH) ? 'x' : '-';
//硬链接数
int linkNum = st.st_nlink;
//文件所有者
char * fileUser = getpwuid(st.st_uid)->pw_name;
//文件所在组
char * fileGrp = getgrgid(st.st_gid)->gr_name;
//文件大小
long int fileSize = st.st_size;
//获取修改的时间
char * time = ctime(&st.st_mtime);
char mtime[512] = {0};
//拷贝函数
strncpy(mtime, time, strlen(time) - 1);
char buf[1024];
sprintf(buf, "%s %d %s %s %ld %s %s", perms, linkNum, fileUser, fileGrp, fileSize, mtime, argv[1]);
printf("%s %d %s %s %ld %s %s\n", perms, linkNum, fileUser, fileGrp, fileSize, mtime, argv[1]);
return 0;
}