总结open与fopen的区别

本文详细对比了C语言中的open与fopen函数,从来源、移植性、适用范围、文件IO层次及缓冲机制等方面进行了分析。帮助读者理解两者的不同应用场景,提升编程效率。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

https://www.zybuluo.com/yiltoncent/note/87461

参考链接1 
参考链接2

对于这两个名字很类似的函数,对于很多初学者来说,不容易搞清楚它们有什么不同,只知道按照函数用法使用。如果能很好的区分两者,相信大家对于C语言和UNIX系统(包括LINUX)有更深入的了解。

在网上查找了一些资料,但是感觉不够全面,一些答案只是从某个角度阐述,所以让人觉得,这个也对,那个也对。但到底谁的表述更正确呢?其实都是对的,只是解释的视角不同罢了。下面结合个人的理解做一些梳理。

 

1. 来源

从来源的角度看,两者能很好的区分开,这也是两者最显而易见的区别:

  • open是UNIX系统调用函数(包括LINUX等),返回的是文件描述符(File Descriptor),它是文件在文件描述符表里的索引。
  • fopen是ANSIC标准中的C语言库函数,在不同的系统中应该调用不同的内核api。返回的是一个指向文件结构的指针。 
     PS:从来源来看,两者是有千丝万缕的联系的,毕竟C语言的库函数还是需要调用系统API实现的。
 

2. 移植性

这一点从上面的来源就可以推断出来,`fopen`是C标准函数,因此拥有良好的移植性;而`open`是UNIX系统调用,移植性有限。如windows下相似的功能使用API函数`CreateFile`。

 

3. 适用范围

  • open返回文件描述符,而文件描述符是UNIX系统下的一个重要概念,UNIX下的一切设备都是以文件的形式操作。如网络套接字、硬件设备等。当然包括操作普通正规文件(Regular File)。
  • fopen是用来操纵普通正规文件(Regular File)的。
 

4. 文件IO层次

如果从文件IO的角度来看,前者属于低级IO函数,后者属于高级IO函数。低级和高级的简单区分标准是:谁离系统内核更近。低级文件IO运行在内核态,高级文件IO运行在用户态。

 

5. 缓冲

  1. 缓冲文件系统 
    缓冲文件系统的特点是:在内存开辟一个“缓冲区”,为程序中的每一个文件使用;当执行读文件的操作时,从磁盘文件将数据先读入内存“缓冲区”,装满后再从内存“缓冲区”依此读出需要的数据。执行写文件的操作时,先将数据写入内存“缓冲区”,待内存“缓冲区”装满后再写入文件。由此可以看出,内存“缓冲区”的大小,影响着实际操作外存的次数,内存“缓冲区”越大,则操作外存的次数就少,执行速度就快、效率高。一般来说,文件“缓冲区”的大小随机器 而定。fopen, fclose, fread, fwrite, fgetc, fgets, fputc, fputs, freopen, fseek, ftell, rewind等。
  2. 非缓冲文件系统 
    缓冲文件系统是借助文件结构体指针来对文件进行管理,通过文件指针来对文件进行访问,既可以读写字符、字符串、格式化数据,也可以读写二进制数据。非缓冲文件系统依赖于操作系统,通过操作系统的功能对文件进行读写,是系统级的输入输出,它不设文件结构体指针,只能读写二进制文件,但效率高、速度快,由于ANSI标准不再包括非缓冲文件系统,因此建议大家最好不要选择它。open, close, read, write, getc, getchar, putc, putchar等。

一句话总结一下,就是open无缓冲,fopen有缓冲。前者与readwrite等配合使用, 后者与fread,fwrite等配合使用。

使用fopen函数,由于在用户态下就有了缓冲,因此进行文件读写操作的时候就减少了用户态和内核态的切换(切换到内核态调用还是需要调用系统调用API:readwrite);而使用open函数,在文件读写时则每次都需要进行内核态和用户态的切换;表现为,如果顺序访问文件,fopen系列的函数要比直接调用open系列的函数快;如果随机访问文件则相反。

这样一总结梳理,相信大家对于两个函数及系列函数有了一个更全面清晰的认识,也应该知道在什么场合下使用什么样的函数更合适,效率更高。

转载于:https://www.cnblogs.com/NickyYe/p/5497659.html

### 文件操作函数 `fopen` `open` #### 使用 `fopen` 打开文件 `fopen` 是标准库中的一个函数,用于打开流模式下的文件。此函数返回指向 `FILE*` 类型的指针。 ```c #include <stdio.h> FILE *fopen(const char *path, const char *mode); ``` 参数说明: - `path`: 要打开的文件路径名字符串。 - `mode`: 指定访问模式的字符串,如 `"r"` 表示只读,`"w"` 表示写入(如果文件存在则清空),`"a"` 表示追加等[^1]。 #### 使用 `open` 打开文件 相比之下,`open` 属于 POSIX 标准的一部分,在 Unix/Linux 系统调用层面工作。它不依赖于 C 的 I/O 库而是直接通过操作系统接口来处理文件描述符。 ```c #include <fcntl.h> #include <unistd.h> int open(const char *pathname, int flags); // 或者带有可选第三个参数 mode_t mode 当创建新文件时设置权限位 int open(const char *pathname, int flags, mode_t mode); ``` 这里, - `pathname`: 同样是指向要打开或创建之文件名称的指针; - `flags`: 定义了如何对待该文件的一系列标志组合,比如 O_RDONLY(只读),O_WRONLY(仅写),O_RDWR(读/写)以及其它控制选项像 O_CREAT 如果不存在就新建文件等等; - `mode`: 只有当设置了 O_CREAT 并且确实需要创建一个新的文件的时候才会被用来指定其权限属性[^2]。 #### 主要差异点总结 | 特性 | `fopen()` | `open()` | |--|--------------------------------------| | 返回值 | FILE 结构体指针 | 整数形式的文件描述符 | | 错误检测 | NULL | -1 | | 编程层次 | 高级I/O 流 | 低级别的系统调用 | | 功能特性 | 支持缓冲区管理 | 提供更细粒度的操作 | 对于大多数应用程序来说,除非特别需求,通常推荐使用更高层面上的标准C库函数 `fopen()`, 因为其跨平台兼容性和易于使用的API设计使得开发更加简便高效;而 `open()` 则更适合那些对性能敏感的应用场景或者是需要利用特定Unix特性的场合[^3]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值