大多数UNIX应用程序都使用I/O库,本章说明了该库所包含的所有函数,以及某些实现细节和效率方面的考虑。同时需要重点关注标准I/O使用了缓冲的技术,但同时也是因为它的出现,产生了很多细节上的问题.

流和FILE对象

unix系统调用的函数都是针对文件描述符操作的.而标准I/O库,它们的操作则是围绕流进行的。当用标准I/O库打开或创建一个文件时,使一个流与一个文件相关联.

关于流定向的问题,当一个流刚被创建时,它并没有定向,我们可以在未定向的流上使用一个多字节I/O函数或者单字节的函数.

#include <stdio.h>
#include <wchar.h> int fwide(FILE* fp,int mode);

当我们打开一个流时,标准I/O函数fopen返回的是一个指向FILE对象的指针.该对象是一个结构,它包含了:用于实际I/O的文件描述符、指向用于该流缓冲区的指针、缓冲区的长度、当前在缓冲区的字符数以及出错标志等。

  

缓冲

标准I/O库提供缓冲的目的是为了尽可能减少使用read和write调用的次数,它也对每个I/O流自动地进行缓冲管理,从而避免了应用程序需要考虑带来的麻烦.标准I/O提供了三种缓冲:

  • 全缓冲:在填满标准I/O缓冲区后才进行实际I/O操作.对于驻留在磁盘上的文件通常是全缓冲的.
  • 行缓冲:当在输入和输出中遇到换行符时,标准I/O库执行I/O操作.这允许我们一次输出一个字符,但只有在写额一行之后才进行实际I/O操作。当流涉及一个终端时(如标准输入和标准输出),通常使用行缓冲.
  • 不带缓冲:标准I/O库不对字符进行缓冲存储。通常标准出错流stderr是不带缓冲的.这就使得出错信息可以尽快显示出来.

对于任何一个给定的流,我们可以自己设定缓冲类型

void setbuf(FILE* fp,char* buf);
int setvbuf(FILE* fp,char* buf,int mode,size_t size);
//返回值:若成功则返回0,若出错则返回非0值

具体参数:setbuf函数

关闭缓冲:buf=NULL

setvbuf 具体参数:mode参数:_IOFBF 全缓冲  _IOLBF 行缓冲 _IONBF 不带缓冲

如果指定全缓冲或行缓冲,则buf和size可选择地指定一个缓冲区及其长度.

打开与关闭流:

#include <stdio.h>

FILE* fopen(const char* pathname,const char* type);//打开一个指定的文件
FILE* freopen(const char* pathname,const char* type,FILE* fp);//用于在一个指定的流上打开一个指定的文件.
FILE* fdopen(int filedes,const char* type);//获取一个现有的文件描述符,并使一个标准的IO流与该描述符相结合.
int fclose(FILE* fp);//关闭文件流

读写流:

一次一个字符的I/O.

#include <stdio.h>
int getc(FILE* fp);
int fgetc(FILE* fp);
int getchar(void);
//若成功则返回下一个字符,若已到达文件尾或出错则返回EOF /*******对应的输出函数***********/
int putc(int c,FILE* fp);
int fputc(int c,FILE* fp);
int putchar(int c);

一次一行字符的I/O

#include <stdio.h>

char* fgets(char* buf,int n,FILE* fp);
char* gets(char* buf);
//成功则返回buf,若已到达文件尾或者出错则返回NULL int fputs(const char* str,FILE* fp);
int puts(const char* str);
//若成功则返回非负值,若出错则返回EOF

注意点:

对于fgets,必须指定缓冲区的长度n,此函数一直读到下一个换行符为止,但是不超过n-1个字符,读入的字符被送入缓冲区.该缓冲区以null结尾。而gets是一个不安全的函数,它可能会造成缓冲区溢出.同时它与fgets的另一个区别是gets并不将换行符存入缓冲区中.

二进制I/O

#include <stdio.h>
size_t fread(void* ptr,size_t size,size_t obj,FILE* fp);
size_t fwrite(const void* ptr,size_t size,size_t obj,FILE* fp);
//读或写的对象数

通常用于读或写一个二进制数组或者用于读写一个结构体.

格式化I/O:

#include <stdio.h>

int printf(const char* format,...);
int fprintf(FILE* fp,const char* format,...);//若成功则返回输出的字节数,若输出出错则返回负值.
int sprintf(char* buf,const char* format,...);
int snprintf(char* buf,size_t n,const char* format,...);
//若成功则返回存入数组的字符数,若编码出错则返回负值. int scanf(const char* format,....);
int fscanf(FILE* fp,const char* format,...);
int sscanf(const char* buf,cpnst char* format,...);
//指定的输入项数;若输入出错或在任意变换前已到达文件尾则返回EOF.

定位流:

#include <stdio.h>

long ftell(FILE* fp);//若成功则返回当前文件位置指示,若出错则返回-1L.
int fseek(FILE* fp,long offset,int whence);//若成功则返回0,,若出错则返回非0值 void rewind(FILE* fp);

临时文件:

#include <stdio.h>

char* tmpnam(char* ptr);//指向唯一路径名的指针
FILE* tmpfile(void);//若成功则返回文件指针,若出错则返回NULL. char* tempnam(const char* directory,const char* prefix);//返回指向唯一路径名的指针.

示例:

#include <stdio.h>

char* tmpnam(char* ptr);//指向唯一路径名的指针
FILE* tmpfile(void);//若成功则返回文件指针,若出错则返回NULL. #include <stdio.h> int main(void){
char name[L_tmpnam],line[MAX_LINE];
FILE *fp; printf("%s\n",tmpnam(NULL));//first tmp name tmpnam(name);
printf("%s\n",name); if((fp=tmpfile())==NULL){
printf("tmpfile error");
exit(-);
} fputs("one line of output\n",fp);
rewind(fp); if(fgets(line,MAX_LINE,fp)==NULL){
printf("fgets error\n");
exit(-);
} fputs(line,stdout);
exit();
}

《APUE》-第五章标准IO库的更多相关文章

  1. C++ Primer 读书笔记: 第8章 标准IO库

    第8章 标准IO库 8.1 面向对象的标准库 1. IO类型在三个独立的头文件中定义:iostream定义读写控制窗口的类型,fstream定义读写已命名文件的类型,而sstream所定义的类型则用于 ...

  2. 《C++ Primer 4th》读书笔记 第8章-标准IO库

    原创文章,转载请注明出处:http://www.cnblogs.com/DayByDay/p/3936457.html

  3. c++ primer 学习杂记3【标准IO库】

    第8章 标准IO库 发现书中一个错误,中文版p248 流状态的查询和控制,举了一个代码例子: int ival; // read cin and test only for EOF; loop is ...

  4. [APUE]标准IO库(下)

    一.标准IO的效率 对比以下四个程序的用户CPU.系统CPU与时钟时间对比 程序1:系统IO 程序2:标准IO getc版本 程序3:标准IO fgets版本 结果: [注:该表截取自APUE,上表中 ...

  5. [APUE]标准IO库(上)

    一.流和FILE对象 系统IO都是针对文件描述符,当打开一个文件时,即返回一个文件描述符,然后用该文件描述符来进行下面的操作,而对于标准IO库,它们的操作则是围绕流(stream)进行的. 当打开一个 ...

  6. C5 标准IO库:APUE 笔记

    C5 :标准IO库 在第三章中,所有IO函数都是围绕文件描述符展开,文件描述符用于后续IO操作.由于文件描述符相关的操作是不带缓冲的IO,需要操作者本人指定缓冲区分配.IO长度等,对设备环境要求一定的 ...

  7. 文件IO函数和标准IO库的区别

    摘自 http://blog.chinaunix.net/uid-26565142-id-3051729.html 1,文件IO函数,在Unix中,有如下5个:open,read,write,lsee ...

  8. 高级UNIX环境编程5 标准IO库

    标准IO库都围绕流进进行的 <stdio.h><wchar.h> memccpy 一般用汇编写的 ftell/fseek/ftello/fseeko/fgetpos/fsetp ...

  9. 18、标准IO库详解及实例

    标准IO库是由Dennis Ritchie于1975年左右编写的,它是Mike Lestbain写的可移植IO库的主要修改版本,2010年以后, 标准IO库几乎没有进行什么修改.标准IO库处理了很多细 ...

随机推荐

  1. Zepto源码笔记(一)

    最近在研究Zepto的源码,这是第一篇分析,欢迎大家继续关注,第一次写源码笔记,希望大家多指点指点,第一篇文章由于首次分析原因不会有太多干货,希望后面的文章能成为各位大大心目中的干货. Zepto是一 ...

  2. Flash AIR 保存与读取

    package { import flash.display.Sprite; import flash.net.URLLoader; import flash.display.Loader; impo ...

  3. 安卓,通过本地应用分享到微信、facebook等

    别的不说了,直接上代码. 支持分享到微信.微博.facebook.twitter package com.example.shareSample; import java.util.List; imp ...

  4. CoreData Multiple Context性能分析-读书笔记

    From: http://floriankugler.com/blog/2013/4/29/concurrent-core-data-stack-performance-shootout  http: ...

  5. poj 2392 Space Elevator(多重背包+先排序)

    Description The cows are going to space! They plan to achieve orbit by building a sort of space elev ...

  6. javascript对象的理解

    从代码中体会javascript中的对象: <!DOCTYPE html> <html> <head> <meta charset="utf-8&q ...

  7. c语言10个经典小程序

    [程序1] 题目:有1.2.3.4个数字,能组成多少个互不相同且无重复数字的三位数?都是多少? 1.程序分析:可填在百位.十位.个位的数字都是1.2.3.4.组成所有的排列后再去 掉不满足条件的排列. ...

  8. Cortex-A9 UART

    一.Exynos4412 UART 的特性 Exynos4412 中UART,有4 个独立的通道,每个通道都可以工作于中断模式或DMA 模式,即 UART 可以发出中断或 DMA 请求以便在UART ...

  9. Nginx的启动脚本

    vi /etc/init.d/nginx #!/bin/sh # nginx        Startup script for nginx # chkconfig: - 85 15 # proces ...

  10. CentOS 恢复 rm -rf * 误删数据(转)

    一. 将磁盘分区挂载为只读 这一步很重要,并且在误删除文件后应尽快将磁盘挂载为只读.越早进行,恢复的成功机率就越大. 1.  查看被删除文件位于哪个分区 [root@localhost  ~]# mo ...