形式:#include
ssize_t read (int filedes, void *buf, size_t nbytes );
成功:返回读到的字节数;出错:返回-1;文件尾:返回0;
原因:基本系统调用功能;
实现:文件(由filedes所指)-读nbytes字节->内存buf中。
补充:有多种情况可使实际读到的字节数少于要求读的字节数:
当从普通文件读时,在读到要求字节数之前已到达了文件尾端。
当从终端设备读时,通常一次最多读一行。
当从网络读时,网络中缓冲机构可能造成返回值小于所要求读的字节数。
当从管道或FIFO读时,如若管道包含的字节少于所需的数量,那么只返回实际用的字节数。
当从某些面向记录的设备读时,一次最多返回一个记录。
当某一信号造成中断,而已经读了部分数据量时。
读操作从文件的当前偏移量处开始,在成功返回之前,该偏移量将增加实际读到的字节数。常用的unix系统shell都提供一种方法,它在标准输入上打开一个文件,在标准输出上追寻或重写一个文件,这使得程序不必自行打开输入和输出文件。
形式:#include
ssize_t write (int filedes, const void *buf, size_t nbytes );
成功:返回已写的字节数;出错:返回-1;
原因:基本系统调用功能;
实现:文件(由filedes所指)<-写nbytes字节-内存buf中。
补充:write出错的一个常见的原因是:磁盘已写满,或者超过了一个给定进程的文件长度限制。对于普通文件,写操作从文件的当前偏移量处开始。如果在打开该文件时,指定了O_APPEND选项,则在每次写操作之前,将文件偏移量设置在文件的当前结尾处。在一次成功写之后,该文件偏移量增加实际写的字节数。
形式:#include
ssize_t pread (int filedes, void *buf, size_t nbytes, off_t offset );
成功:返回读到的字节数;出错:返回-1;到文件结尾:返回0
原因:由于lseek和read 调用之间,内核可能会临时挂起进程,所以对同步问题造成了问题,调用pread相当于顺序调用了lseek 和 read,这两个操作相当于一个捆绑的原子操作。
实现:文件(由filedes所指)-读nbytes字节->内存buf中。
补充:调用pread时,无法中断其定位和读操作,另外不更新文件指针。
形式:#include
ssize_t pwrite (int filedes, const void *buf, size_t nbytes, off_t offset );
成功:返回已写的字节数;出错:返回-1;
原因:由于lseek和write 调用之间,内核可能会临时挂起进程,所以对同步问题造成了问题,调用pwrite相当于顺序调用了lseek 和 write,这两个操作相当于一个捆绑的原子操作。
实现:文件(由filedes所指)<-写nbytes字节-内存buf中。
补充:调用pwrite时,无法中断其定位和读操作,另外不更新文件指针。
2. 流(stream)或标准I/O( 进程->fp->流(FILE+缓冲)->文件)(内存buf, 流fp):
每次输入一个字符:
格式:#include
int getc(FILE *fp);
成功:返回下一个字符;出错:返回EOF;文件尾:EOF;
实现:内存 <-读一个字符c- 流(由fp所指的流,是文件的逻辑代表)
原因:在标准I/O中用,将流看成文件的逻辑代表,将对进程->文件的操作,现转换为进程->流(也就是相当于文件)的操作。
补充:函数在返回下一个字符时,会将其unsigned char类型转换为int类型。为不带符号的理由是,如果最高位是1也不会使返回值为负。要求整形返回值的理由是,这样就可以返回所有可能的字符值再加上一个已出错或已到达文件尾端的指示值。即字符值变为正的int值,负的值就是出错或是到达文件尾端。(负值表特殊意义),同时不论是出错还是到达文件尾端,这三个函数都返回同样的值即都是-1。由于每个流在FILE对象中维持了两个标志,即出错标志和文件结束标志,为了区分其不同,必须调用ferror或feof。
格式:#include
int fgetc(FILE *fp);
成功:返回下一个字符;出错:返回EOF;文件尾:EOF;
实现:同getc
原因:同getc
补充:同getc
格式:#include
int getchar(void);
成功:返回下一个字符;出错:返回EOF;文件尾:EOF;
实现:内存 <-读一个字符c- 流(由stdin所指的流,是标准输入文件的逻辑代表),所以getchar=getc(stdin);
原因:同getc
补充:同getc
每次输入一行:
格式:#include
char *fgets(char *restrict buf, Int n, FILE *restrict fp);
成功:返回buf;出错:返回NULL; 文件结尾:NULL;
实现:内存buf <-从fp所指的流中取一行字符- 流(由fp所指)
原因:在标准I/O中用,将流看成文件的逻辑代表,将对进程->文件的操作,现转换为进程->流(也就是相当于文件)的操作。
补充:必须指定用户进程缓冲区的长度n,即buf的大小,此函数从流中一直读到下一个换行符为止,但是不超过n-1个字符,读入的字符被送入用户缓冲区buf中。该缓冲区以null字符结尾。如若该行包括最后换行符的字数大于n-1,则其只返回一个不完整的行,但是缓冲区buf总是以null字符结尾,对此函数的调用会继续读该行。缓冲区buf中的内容为:(字符+换行符)+null。所以字符+换行符<=n-1,因为一定要留一个NULL字符来标识缓冲区的结束;
格式:#include
char *gets(char * buf);
成功:返回buf;出错:返回NULL; 文件结尾:NULL;
实现:内存buf <-从stdin所指的流中取1行字符-标准输入流(由fp=stdin所指)
原因:同上;
补充:不推荐使用,问题是调用者在使用gets时,不能指定缓冲区buf(用户进程)的长度,这样可能造成缓冲区溢出。
每次输出一个字符:
格式:#include
int putc(int c ,FILE *fp);
成功:返回c;出错:返回EOF;
实现:内存中整形变量c-写字符C->流(由fp所指)。至于流什么时候将C写入文件中,这个由库函数来实现,不用用户操心;
格式:#include
int fputc(int c ,FILE *fp);
成功:返回c;出错:返回EOF;
实现:内存中整形变量c-写字符C->流(由fp所指)。至于流什么时候将C写入文件中,这个由库函数来实现,不用用户操心;
格式:#include
int putchar(int c);
成功:返回c;出错:返回EOF;
实现:内存中整形变量c-写字符C->流(由fp=stdout所指)。至于流什么时候将C写入标准输出文件中,这个由库函数来实现,不用用户操心;
补充:putchar(c)=putc(c,stdout);
每次输出一行:
格式:#include
int fputs(const char *restrict str, FILE *restrict fp);
成功:返回非负值;出错:返回EOF;
实现:内存中字符数组str-写字符数组str->流(由fp所指)。
补充:将一个以null符终止的字符串(相当于用户空间buf,肯定有null,对应于fgets的buf中一定要有个null来标识缓冲区buf的结束。)写到指定的流,尾端的终止符null不写进流中。注意,这并不一定是每次输出一行,因为它并不要求在null之前一定是换行符,buf中有就有,没有就没有,通常,在空字符之前是一个换行符,但并不要求总是如此。用户空间buf:字符(+换行符)+null;流中的buf:字符+换行符。
格式:#include
int puts(const char * str);
成功:返回非负值;出错:返回EOF;
实现:内存中字符数组str-写字符数组str->标准输出流(由fp=stdout所指)。
补充:将一个以null结尾的字符串写到标准输出上,相当于进程->流->标准输出文件。终止符不写出,但是puts然后又将一个换行符写到标准输出。应当少用,以免需要记住它在最后是否添加了一个换行符。而fgets和fputs在处理换行符,本着实事求是的态度,有就有,没有就没有,不会在用户buf和流缓冲以及文件中自己添加,只是在数据经过流缓冲时,增加或是过滤到null字符。当fgets时会在用户buf中增加一个null以标识用户buf的结束,而fputs时,以null为终止字符,但是尾端的null并不写在流中。
二进制I/O:
格式:#include
ssize_t fread(void *restrict ptr, size_t size, size_t nobj, FILE *restrict fp);
成功:读到的对象数。
实现:内存始址ptr<-读N个对象- 流(由fp所指)
原因:以上有一次一个字符或是一次一行的方式进行I/O操作,当我们读或写一个结构时,对于一次一个字符的方式,必须循环通过整个结构,每次循环处理一个字节,一次读或写一个字节,这会很烦。而对于一次一行的方式,当每次结构体中有null字符时,fputs就会停止,所以也不能用它实现读结构,同时fgets中包含有null字节或换行符,其也不能正常工作。所以要并实现结构体作为一个整体的读或写。
补充:使用二进制的基本问题是:它只能用于读在同一系统上已写的数据。其原因是:在结构中,同一成员偏移量可能因为编译器和系统而异,另外,用来存储多字节整数和浮点值的二进制格式在不同的机器体系结构之间也可能不同。
格式:#include
ssize_t fwrite(const void *restrict ptr, size_t size, size_t nobj, FILE *restrict fp);
成功:写的对象数。
实现:内存始址ptr-写N个对象-> 流(由fp所指)
格式化输入:文件-流->格式转换->内存变量中
格式:#include
int scanf(const char *restrict format,…)
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF;
实现:标准输入流->格式转换->内存变量中。用于分析输入字符串,并将字符序列转换成指定类型的变量。格式之后的各个参数包含了变量的地址,以用转换结果初始化这些变量。
原因:要在流中做格式转换,再将结果放到内存变量中
格式:#include
int fscanf(FILE *restrict fp, const char *restrict format,…)
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF;
实现:输入流->格式转换->内存变量中
格式:#include
int sscanf(const char *restrict buf, const char *restrict format,…)
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF;
实现:内存buf->格式转换->内存变量中。
补充:对于scanf(), 从标准输入流中输入;fscanf,从流中输入; sscanf,这个比较特殊,不是从流中输入,而是内存的一个buf相当于string中输入。
格式:#include
int vscanf(const char *restrict format, va_list arg);
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF
实现:标准输入流->格式转换->内存变量中。用于分析输入字符串,并将字符序列转换成指定类型的变量。格式之后的各个参数包含了变量的地址,以用转换结果初始化这些变量。同于scanf,只是将原来的可变参数…换成了arg;
原因:要在流中做格式转换,再将结果放到内存变量中
格式:#include
int vfscanf(FILE *restrict fp, const char *restrict format, va_list arg)
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF;
实现:输入流->格式转换->内存变量中, 同于fscanf,只是将原来的可变参数…,换成了arg;
格式:#include
int vsscanf(const char *restrict buf, const char *restrict format, va_list arg)
成功:指定的输入项数;出错:返回EOF;输入出错或在任意变换前已到达文件结尾:EOF;
实现:内存buf->格式转换->内存变量中。同于sscanf,只是将原来的可变参数…,换成了arg;
补充:对于scanf(), 从标准输入流中输入;fscanf,从流中输入; sscanf,这个比较特殊,不是从流中输入,而是内存的一个buf相当于string中输入。
格式化输出:文件-流<-格式字符串<-内存变量
格式:#include
int printf(const char *restrict format, …);
成功:返回输出字符数;出错:返回负值;
实现:标准输出流<-格式字符串<-内存变量
原因:要将内存变量的数据做格式变换,再将变换的结果放入流中
格式:#include
int fprintf(FILE *restrict fp,const char *restrict format, …);
成功:返回输出字符数;出错:返回负值;
实现:文件-输出流<-格式字符串<-内存变量
格式:#include
int sprintf(char *restrict buf, const char *restrict format, …);
成功:返回输出字符数;出错:返回负值;
实现:内存字符串buf<-格式字符串<-内存变量,就是将格式化的字符串送入数组buf而不是指定的流中。在数组的尾端自动加一个null字节,但该字节不包括在返回值中。
格式:#include
int snprintf(char *restrict buf, size_t n , const char *restrict format, …);
成功:返回输出字符数;出错:返回负值;
实现:内存字符串buf<-格式字符串<-内存变量,就是将格式化的字符串送入数组buf而不是指定的流中。在数组的尾端自动加一个null字节,但该字节不包括在返回值中。只能输入n-1个字符,超过的任何字条都会被丢弃。
格式:#include
#include
int vprintf(const char *restrict format, va_list arg);
成功:返回输出字符数;出错:返回负值;
实现:标准输出流<-格式字符串<-内存变量,同于printf,只是将原来的可变参数…换成了arg;
原因:要将内存变量的数据做格式变换,再将变换的结果放入流中
格式:#include
#include
int vfprintf(FILE *restrict fp,const char *restrict format, va_list arg);
成功:返回输出字符数;出错:返回负值;
实现:输出流<-格式字符串<-内存变量,同于fprintf,只是将原来的可变参数…换成了arg;
原因:要将内存变量的数据做格式变换,再将变换的结果放入流中
格式:#include
#include
int vsprintf(char *restrict buf, const char *restrict format, va_list arg);
成功:返回输出字符数;出错:返回负值;
实现:内存数组buf<-格式字符串<-内存变量,同于sprintf,只是将原来的可变参数…换成了arg; 就是将格式化的字符串送入数组buf而不是指定的流中。在数组的尾端自动加一个null字节,但该字节不包括在返回值中。
原因:要将内存变量的数据做格式变换,再将变换的结果放入流中
格式:#include
int vsnprintf(char *restrict buf, size_t n , const char *restrict format, va_list arg);
成功:返回输出字符数;出错:返回负值;
实现:内存字符串buf<-格式字符串<-内存变量, 同于snprintf,只是将原来的可变参数…换成了arg; 就是将格式化的字符串送入数组buf而不是指定的流中。在数组的尾端自动加一个null字节,但该字节不包括在返回值中。只能输入n-1个字符,超过的任何字条都会被丢弃。
格式:#include
ssize_t readv(int filedes, const struct iovec *iov, int iovcnt);
成功:返回已读的字节数;出错:返回-1;
实现:文件(fd)->内存向量中
原因:在一次函数调用中读、写多个非连续缓冲区,但是这些缓冲区已经用iovec表示好了。减少了系统调用的次数。
格式:#include
ssize_t writev(int filedes, const struct iovec *iov, int iovcnt);
成功:返回已读的字节数;出错:返回-1;
实现:文件(fd)<-内存向量
原因:在一次函数调用中读、写多个非连续缓冲区,但是这些缓冲区已经用iovec表示好了。减少了系统调用的次数。
格式:#include
ssize_t readn(int filedes, void *bug, size_t nbytes);
成功:返回已读的字节数;出错:返回-1;
实现:文件(fd)->内存buf中
原因:管道、FIFO以及某些设备,特别是终端、网络和STREAMS设备有下列两种性质:一是,一次read操作所返回的数据可能少于所要求的数据,即使还没达到文件尾端也可能是这样的。这不是一个错误,应当继续读该设备。二是,一次write操作所返回的值也可能少于所指定输出的字节数,这可能是由若干因素造成的。这些也不是错误,也应当继续写余下的数据至该设备。通常只对非阻塞描述符,或捕捉到一个信号时,才发生这种write的中途返回。但是在读写磁盘时,很少遇到这样的情况。所以这个函数其实是按需要多次调用read 和write直至读、写了N个字节数据,即我们称之为:直到集齐了再返回。
格式:#include
ssize_t writen(int filedes, void *bug, size_t nbytes);
成功:返回已读的字节数;出错:返回-1;
实现:文件(fd)<-内存buf中
原因:管道、FIFO以及某些设备,特别是终端、网络和STREAMS设备有下列两种性质:一是,一次read操作所返回的数据可能少于所要求的数据,即使还没达到文件尾端也可能是这样的。这不是一个错误,应当继续读该设备。二是,一次write操作所返回的值也可能少于所指定输出的字节数,这可能是由若干因素造成的。这些也不是错误,也应当继续写余下的数据至该设备。通常只对非阻塞描述符,或捕捉到一个信号时,才发生这种write的中途返回。但是在读写磁盘时,很少遇到这样的情况。所以这个函数其实是按需要多次调用read 和write直至读、写了N个字节数据,即我们称之为:直到集齐了再返回。