一.标准IO库接口的介绍:
1.标准库IO接口的分类:
(1)fopen:打开文件;
接口:FILE *fopen(const char *path, const char *mode);
参数:
path:文件路径;
mode:文件打开方式;
r:只读方式打开文件,读的位置在文件的起始位置;
r+:可读可写方式打开文件,文件的读写在文件的起始位置;
w:只写方式打开文件,若文件不存在则创建,若文件存在则清空原有内容;
w+:可读可写方式打开文件,若文件不存在则创建,若文件存在则清空原有内容;
a:追加只写,写的时候写到文件的末尾,若文件不存在则创建;
a+:可读可写;多出了可读操作,读的时候在文件的起始位置,写的时候在文件的末尾;
(2)fread和fwrite:读操作和写操作;
fread: 接口:size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
参数:
ptr:用于接收数据缓冲区;
size:要读取的块的大小;
nmemb:要读取的块的个数;
stream:文件流指针;
返回值:实际读取的块的个数; 若读到文件末尾则返回0;
fwrite: 接口:size_t fwrite(const void *ptr, size_t size, size_t nmemb,FILE *stream);
参数:
ptr:要写入数据缓冲区;
size:要写入的块的大小;
nmemb:要写入放入块个数;
stream:fopen返回的文件轮指针;
返回值:实际写入的块个数;
(3)fseek:跳转读写位置;
接口:int fseek(FILE *stream, long offset, int whence);
对stream文件的读写位置跳转到whence位置偏移offset个字节处;
参数:
whence:跳转的起始位置;
SEEK_SET:文件的起始位置;
SEEK_CUT:文件的读写位置;
SEEK_END:文件的末尾位置;
offset:偏移量;
(4)fclose:关闭文件;
接口:int fclose(FILE *fp);
(5)fget:获取字符串函数;
从文件结构体指针stream中读取数据,每次读取一行。读取的数据保存在buf指向的字符数组中,每次最多读取bufsize-1个字符;
接口:char *fgets(char *buf, int bufsize, FILE *stream);
参数:
*buf: 字符型指针,指向将存储到的数据地址;
bufsize: 整型数据,指明buf指向的字符数组的大小;
*stream: 文件结构体指针,将要读取的文件流;
2.具体代码实现原理:
#include <unistd.h>
#include <stdio.h>
#include <errno.h>
#include <string.h>
int main(){
FILE *fp=fopen("./tmp.txt","w+"); //创建文件;
if(fp==NULL){
perror("fopen error");
return -1;
}
char *ptr="hello world~~\n";
size_t ret=fwrite(ptr,strlen(ptr),1,fp); //写入文件;
if(ret==0){
printf("fopen error\n");
return -1;
}
fseek(fp,0,SEEK_SET); //跳转读写位置,到文件的起始位置;
char buf[1024]={0};
ret=fread(buf,1023,1,fp); //读文件内容;
printf("ret:%d-[%s]\n",ret,buf);
//关闭文件;
//int fclose(FILE *fp);
fclose(fp);
return 0;
}
输出结果:
二.系统调用接口:
1.系统调用接口的分类:
(1)open:打开文件;
接口:
int open(const char *pathname, int flags, mode_t mode);
参数:
pathname:指定要打开的文件名称;
flags:选项参数;
其中分为必选参数和可选参数;
1.必选参数:
O_RDONLY(以只读方式打开文件)
O_WRONLY(以只写方式打开文件)
O_RDWR(以读写方式打开文件)
只能选择其一,不能全部使用;
2.可选参数:
O_CREAT(若文件存在则打开,否则创建新文件)
O_EXCL (与O_CREAT同时使用,若文件存在则报错;不存在则创建)
O_TRUNC(打开文件的同时截断文件为0长度)
O_APPEND(写数据的时候总是写到文件末尾)
返回值:
成功:返回文件描述符(正整数); 失败;-1;
(2)read和write:读写文件;
read接口: ssize_t read(int fd, void *buf, size_t count);
参数:
fd:文件描述符;
buf:读出数据的缓冲区;
count:为每次读取的字节数(是请求读取的字节数,读上来的数据保存在缓冲区buf中,同时文件的当前读写位置向后移)
功能:从fd文件中读取count长度的数据,放到buf中;
返回值:成功:返回实际读取到的字节数; 失败:-1;
write接口: ssize_t write(int fd, const void *buf, size_t count);
参数:
fd:open打开文件所返回的描述符(write所对应的是写,即就是1);
buf:通常是一个字符串,需要写入的字符串;
count:是每次写入的字节数;
返回值:成功:返回实际写入的字节数; 失败:-1;
(4)lseek:跳转读写位置;
接口:
off_t lseek(int fd, off_t offset, int whence);
功能:跳转到fd文件的读写位置到指定处;
参数:
whence:跳转的起始位置;
SEEK_SET:文件的起始位置;
SEEK_CUT:文件的读写位置;
SEEK_END:文件的末尾位置;
offset:偏移量;
(5)close:关闭文件;
int close(int fd);
2.具体代码实现原理:
#include <errno.h> #include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <sys/types.h>
#include <string.h>
#include <stdlib.h>
int mian(){
umask(0);
int fd=open("./text.txt",O_RDWR| O_CREAT| O_APPEND| O_TRUNC, 0777);
//文件存在则打开,文件不存在则创建文件;
if(fd<0){
perror("open error");
return -1;
}
char* data="hello world\n"; //将数据信息写入文件;
ssize_t ret=write(fd,data,strlen(data));
if(ret<0){
perror("write error");
return -1;
}
lseek(fd,0,SEEK_SET); //跳转到文件的起始位置;
char buf[1024]={0}; //开始读取文件;
ret=read(fd,buf,1023);
if(ret<0){
perror("read error");
return -1;
}
printf("ret:%d-[%s]\n",ret,buf);
close(fd);
return 0;
}
输出结果为:
三.关于库和系统调用的关系;
具体关于操作系统的概念可查看链接:https://blog.csdn.net/DX_Jone/article/details/96383377
对于 fopen,fclose,fread,fwrite,fget 都是C标准库当中的函数,我们称之为库函数(lib);
而open,close,read,write,lseek都属于系统提供的接口,称之为系统调用接口;
库函数是对系统调用接口的一种封装;管理者并不需要直接与被管理者交互进行管理,而是通过对被管理者进行描述,并且将描述信息得当组织起来进行管理;