Linux标准IO库简介

  • 为什么要设计标准I/O库?
    ➢ 直接使用API进行文件访问时,需要考虑许多细节问题,例如:read、write时,缓冲区的大小该如何确定,才能使效率最优。
    ➢ read和write等底层系统调用函数进行输入输出时,在用户态和内核态之间来回切换,每次读出或写入的数据量较少,导致频繁的I/O操作,增加了系统开销。
  • 标准I/O库是ANSI C规范的一部分,函数原型在文件stdio.h中定义,对底层I/O系统调用进行了封装,为程序员提供了带有格式转换功能的输入输出操作,并在用户空间增加了缓冲区管理。
  • 文件IO和标准IO区别:
  1. 文件I/O:文件I/O称之为不带缓存的IO(unbuffered I/O)。不带缓存指的是每个read,write都调用内核中的一个系统调用。也就是一般所说的低级I/O——操作系统提供的基本IO服务,与os绑定,特定于linix或unix平台。
  2. 标准I/O:标准I/O是ANSI C建立的一个标准I/O模型,是一个标准函数包和stdio.h头文件中的定义,具有一定的可移植性。标准I/O库处理很多细节。例如缓存分配,以优化长度执行I/O等。标准的I/O提供了三种类型的缓存。
    (1)全缓存:当填满标准I/O缓存后才进行实际的I/O操作。
    (2)行缓存:当输入或输出中遇到新行符时,标准I/O库执行I/O操作。
    (3)不带缓存:stderr就是了。
  3. 文件I/O 又称为低级磁盘I/O,遵循POSIX相关标准。任何兼容POSIX标准的操作系统上都支持文件I/O。
  4. 标准I/O被称为高级磁盘I/O,遵循ANSI C相关标准。只要开发环境中有标准I/O库,标准I/O就可以使用。(Linux 中使用的是GLIBC,它是标准C库的超集。不仅包含ANSI C中定义的函数,还包括POSIX标准中定义的函数。因此,Linux 下既可以使用标准I/O,也可以使用文件I/O)。
  5. 通过文件I/O读写文件时,每次操作都会执行相关系统调用。这样处理的好处是直接读写实际文件,坏处是频繁的系统调用会增加系统开销,标准I/O可以看成是在文件I/O的基础上封装了缓冲机制。先读写缓冲区,必要时再访问实际文件,从而减少了系统调用的次数。
  6. 文件I/O中用文件描述符表现一个打开的文件,可以访问不同类型的文件如普通文件、设备文件和管道文件等。而标准I/O中用FILE(流)表示一个打开的文件,通常只用来访问普通文件。
  • 标准IO库
  • 分离了应用程序空间和实际的物理设备
  • 减少了直接读盘次数,提高性能
    ➢ 读取前查看是否已存在页缓存中,如果已经存放在了页缓存中,数据立即返回给应用程序
    ➢ 写数据前先写到页缓存中,如果用户采用的是同步写机制( synchronous writes ), 那么数据会立即被写回到磁盘上,应用程序会一直等到数据被写完为止;如果用户采用的是延迟写机制( deferred writes ),那么应用程序就完全不需要等到数据全部被写回到磁盘,数据只要被写到页缓存中去就可以了。
  • 异步缓存和延迟写机制一样,不等数据全部写回到磁盘,数据只要写到页缓存中即可,不同的是当数据全部写到磁盘上后,异步机制会通知给应用程序,延迟写机制不会。

fopen函数

  • fopen函数功能
    打开一个指定文件
  • 函数原型
    FILE *fopen(const char *restrict pathname, const char *restrict type);
  • 参数
    pathname:要打开的文件名
    type:指定文件的读、写方式
  • type 说明:
  1. r或rb 为读而打开
  2. w或wb 使文件长度为0,或为写而创建
  3. a或ab 添加;为在文件尾写而打开,或为写而创建
  4. r+或r+b或rb+ 为读和写而打开
  5. w+或w+b或wb+ 使文件长度为0,或为读和写而打开
  6. a+或a+b或ab+ 为在文件尾读和写而打开或创建
限制rwar+w+a+
文件必须存在
删除文件以前内容
流可以读
流可以写
流只在尾端处写

setbuf和setvbuf函数

  • setbuf和setvbuf函数功能
    打开和关闭缓冲机制
  • 函数原型
    void setbuf(FILE *steam, char *buf);
    void setvbuf(FILE *steam, char *buf, int mode, size_t size);
函数modebuf缓存及长度缓存的类型
setbufnonnull长度为BUFSIZ的用户缓存全缓存
setbufNULL(无缓存)不带缓存
setvbuf_IOFBFnonnull长度为size的用户缓存全缓存
setvbuf_IOFBFNULL合适长度的系统缓存全缓存
setvbuf_IOLBFnonnull长度为size的用户缓存行缓存
setvbuf_IOLBFNULL合适长度的系统缓存行缓存
setvbuf_IONBF忽略无缓存不带缓存

fdopen函数

  • fdopen函数功能
    取一个现存的文件描述符,并使一个标准I/O流与该描述符相结合
  • 头文件
    #include<stdio.h>
  • 函数原型
    FILE *fdopen(int fd, const char *type);
  • fdopen常用于由创建管道及网络通信通道函数返回的描述符。
    ➢ 这些特殊类型的文件,不能用fopen打开
    ➢ 因此必须先调用设备专用函数以获得一个文件描述符,然后再用fdopen使一个标准I/O流与该描述符相关联
  • 对于fdopen函数,type参数的意义稍有区别
    ➢ 因为该描述符已被打开,所以fdopen为写而打开并不截短该文件
    ➢ 不能用于创建该文件(因为如若一个描述符引用一个文件,则该文件一定已经存在)
Type值操作文件类型是否新建文件是否清空原文件可读可写读写开始位置
r文本文件NONOYESNO文件开头
r+文本文件YESNOYESYES文件开头
w文本文件YESYESNOYES文件开头
w+文本文件YESYESYESYES文件开头
a文本文件NOYESNOYES文件结尾
a+文本文件NOYESYESYES文件结尾
rb二进制文件NONOYESNO文件开头
r+b或rb+二进制文件YESNOYESYES文件开头
wb二进制文件YESYESNOYES文件开头
w+b或wb+二进制文件YESYESYESYES文件开头
ab二进制文件NOYESNOYES文件结尾
a+b或ab+二进制文件NOYESYESYES文件结尾
FILE *fp;
int fd;
if ((fp = fopen("hello.txt", "w+")) == NULL) {
printf("fopen file error\n");
return 0;}
fprintf(fp, "hello word\n");
fclose(fp);
if ((fd = open("hello.txt", O_RDWR)) == -1) {
printf("open file fail\n");
return 0;}
if ((fp = fdopen(fd, "a+")) == NULL) {
printf("fdopen open\n");
return 0;
}
fprintf(fp, "linux c program");
fclose(fp);
      
  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值