news 2026/4/23 13:02:27

linux系统IO

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
linux系统IO

第一阶段:重新认识“文件”

在写 C 语言时,你肯定用过fopen,fread,fwrite。但在操作系统眼里,文件远不止“读写”这么简单。

1. 文件的本质

文件 = 文件内容 + 文件属性

  • 内容:你写进去的 "Hello World"。
  • 属性 (元数据):文件名、大小、创建时间、拥有者、权限等。
  • 结论:创建一个 0kb 的空文件,它也是占磁盘空间的,因为要存它的属性
2. 谁在操作文件?

代码写在那如果不跑,是不会操作文件的。

只有当代码运行起来变成进程后,才是“进程在操作文件”。

所以,文件操作的本质,是 进程 (Process) 和 操作系统 (OS) 之间的一次对话(因为磁盘硬件是归 OS 管的,进程不能直接摸)。


第二阶段:库函数 vs 系统调用 (The Battle)

  • C 标准库函数 (Library Functions)
    • fopen,fclose,fwrite,fread...
    • 特点跨平台。你在 Windows 上写fopen能跑,在 Linux 上也能跑。因为 C 库帮你屏蔽了底层差异。
    • 带缓冲:这是关键!它自带一个用户级缓冲区(稍后详解)。
  • 系统调用接口 (System Calls)
    • open,close,write,read...
    • 特点不跨平台。这是 Linux 内核直接提供的接口(Windows 的 API 叫CreateFile)。
    • 无缓冲:直接把数据扔给内核,甚至直接写盘。

层级关系:


第三阶段:核心接口open详解

我们要重点学习open,因为所有的“黑魔法”(如O_APPEND追加、O_CREAT创建)都藏在它的参数里。

1. 函数原型

你需要包含<fcntl.h>

int open(const char *pathname, int flags); int open(const char *pathname, int flags, mode_t mode);
2. 参数flags(位图标志位)

还记得我们在讲waitpid时提到的位图吗?这里也是一样的设计。Linux 用一个整数的不同比特位来表示不同的选项。

常用的标志(必须记住):

  • O_RDONLY:只读打开。
  • O_WRONLY:只写打开。
  • O_RDWR:读写打开。

(以上三个必须三选一)

  • O_CREAT:如果文件不存在,就创建它。(如果存在,直接打开)。
  • O_TRUNC截断 (Truncate)。如果文件存在,把它清空(长度变为 0)。
  • O_APPEND追加。写数据时自动加到文件末尾。

如何组合? 使用 按位或 (|)。

比如:O_WRONLY | O_CREAT | O_TRUNC 就等同于 C 语言的 fopen(..., "w")。

3. 参数mode(权限)

注意:只有当你使用了O_CREAT选项时,才必须传第三个参数mode

  • 作用:指定新创建文件的初始权限(如0666)。
  • 实际权限:记得我们讲 mkfifo 时说的 umask 吗?这里同理。

$$实际权限 = mode \ \& \ (\sim umask)$$

4. 返回值:文件描述符 (File Descriptor)
  • 成功:返回一个int(大于等于 0)。我们通常叫它fd
  • 失败:返回-1,并设置errno

第四阶段:代码实战 —— 手写fopen("w")

我们来写一段代码,直接使用系统调用来实现“向文件写入字符串”。

#include <stdio.h> #include <string.h> #include <unistd.h> #include <sys/types.h> #include <sys/stat.h> #include <fcntl.h> int main() { // 场景:以写的方式打开,如果不存在就创建,如果存在就清空 // 这完全等价于 fopen("log.txt", "w"); // 设置 umask 为 0,保证我们要的权限不被过滤 umask(0); // 1. 打开文件 // 返回值 fd 就是那个神秘的整数 int fd = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); if (fd < 0) { perror("open"); // 打印错误原因 return 1; } printf("Open success, fd: %d\n", fd); // 2. 写入数据 const char *msg = "hello system call\n"; // write(fd, 缓冲区, 字节数) // 注意:这里不需要 +1 把 '\0' 写进去,因为文件里不需要字符串结束符,那是C语言的规定 int count = 5; while(count--) { write(fd, msg, strlen(msg)); } // 3. 关闭文件 close(fd); return 0; }

编译运行

Bash

gcc sys_io.c -o sys_io ./sys_io cat log.txt

实验现象

  1. 你会看到屏幕打印Open success, fd: 3
  2. cat能看到文件内容。
  3. 关键问题:为什么fd3? 0, 1, 2 去哪了?

第五阶段:核心谜题 —— 文件描述符 (File Descriptor)

这是基础 IO 中最重要的概念,也是面试必考题。

1. 0, 1, 2 的秘密

Linux 进程启动时,默认会打开三个文件:

  • 0 (Standard Input):标准输入(键盘),对应 C 语言的stdin
  • 1 (Standard Output):标准输出(显示器),对应 C 语言的stdout
  • 2 (Standard Error):标准错误(显示器),对应 C 语言的stderr

因为 0, 1, 2 被占用了,所以你新打开的文件自然就分到了3

2. fd 的本质:数组下标

fd 到底是什么?

在内核的 task_struct (PCB) 中,有一个指针指向 struct files_struct。

在这个结构体里,有一个指针数组 struct file* fd_array[]。

  • fd 就是这个数组的下标!
  • 当你调用open时,内核创建一个file结构体,把它填入数组中最小的空闲位置(比如 3),然后把下标 3 返回给你。
  • 当你调用write(3, ...)时,内核通过下标 3 找到对应的file结构体,从而找到文件。
版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/22 21:36:32

[Wf2016]Branch Assignment题解

P6918 [ICPC 2016 WF] Branch Assignment 题目描述 创新消费品公司&#xff08;ICPC&#xff09;计划启动一个绝密项目。该项目由 sss 个子项目组成。将有 b≥sb \ge sb≥s 个 ICPC 的分支机构参与此项目&#xff0c;ICPC 希望将每个分支机构分配给一个子项目。换句话说&#x…

作者头像 李华
网站建设 2026/4/23 11:17:28

谷歌与OpenAI:谁能在AI领域称霸,GPT-5.2与Gemini 3深度对比

2025年&#xff0c;人工智能领域迎来了两大巨头的最新力作&#xff1a;OpenAI 发布了 GPT-5.2&#xff0c;而 谷歌 则推出了 Gemini 3。这两款产品不仅代表了各自公司的技术积淀&#xff0c;也是AI技术的前沿标杆。从语言模型到深度学习应用&#xff0c;GPT-5.2与Gemini 3的发布…

作者头像 李华
网站建设 2026/4/16 17:54:02

jQuery EasyUI 树形菜单 - 创建带复选框的树形菜单

下面直接给你最实用、最常见的创建带复选框的树形菜单&#xff08;Checkbox Tree&#xff09;方法&#xff0c;jQuery EasyUI 的 tree 组件内置支持复选框&#xff0c;复制粘贴就能做出权限分配、部门多选、商品分类批量选择等场景&#xff0c;领导最爱的“勾选父节点自动勾选子…

作者头像 李华
网站建设 2026/4/18 16:01:21

jQuery EasyUI 树形菜单 - 树形菜单拖放控制

下面直接给你最实用、最常见的树形菜单拖放控制&#xff08;Draggable & Droppable Tree&#xff09;方法&#xff0c;jQuery EasyUI 的 tree 组件内置支持拖拽节点&#xff08;移动节点、排序、跨树拖拽&#xff09;&#xff0c;复制粘贴就能用&#xff0c;领导最爱的“菜…

作者头像 李华