【Linux】进程IO|系统调用|open|write|文件描述符fd|封装|理解一切皆文件

简介: 本文详细介绍了Linux中的进程IO与系统调用,包括 `open`、`write`、`read`和 `close`函数及其用法,解释了文件描述符(fd)的概念,并深入探讨了Linux中的“一切皆文件”思想。这种设计极大地简化了系统编程,使得处理不同类型的IO设备变得更加一致和简单。通过本文的学习,您应该能够更好地理解和应用Linux中的进程IO操作,提高系统编程的效率和能力。

理解Linux中的进程IO与系统调用

在Linux操作系统中,进程与系统之间的交互主要通过系统调用完成。文件IO是最常见的系统调用之一,包括打开文件、读写文件等操作。本文将详细介绍Linux中的进程IO、系统调用、文件描述符(fd)及其封装,并深入探讨“理解一切皆文件”的概念。

一、系统调用简介

系统调用(System Call)是操作系统提供给应用程序的编程接口。通过系统调用,应用程序可以请求操作系统提供的各种服务,例如文件操作、进程控制、网络通信等。

在Linux中,常用的文件操作系统调用包括:

  • open:打开文件
  • read:读取文件
  • write:写入文件
  • close:关闭文件

二、文件描述符(File Descriptor)

文件描述符(fd)是一个非负整数,用于标识已打开的文件或其他IO资源。在Linux中,文件描述符是进程级别的,每个进程都有一张独立的文件描述符表。标准文件描述符包括:

  • 0:标准输入(stdin)
  • 1:标准输出(stdout)
  • 2:标准错误(stderr)

三、系统调用详解

3.1 open系统调用

open系统调用用于打开文件,并返回一个文件描述符。其原型定义在 <fcntl.h>头文件中:

#include <fcntl.h>

int open(const char *pathname, int flags, mode_t mode);
​
  • pathname:要打开的文件路径。
  • flags:打开文件的模式(例如 O_RDONLYO_WRONLYO_RDWR)。
  • mode:文件权限(用于创建文件时)。

示例代码:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);
    if (fd == -1) {
        perror("open");
        return 1;
    }
    printf("File opened with fd: %d\n", fd);
    close(fd);
    return 0;
}
​

3.2 write系统调用

write系统调用用于向文件写入数据。其原型定义在 <unistd.h>头文件中:

#include <unistd.h>

ssize_t write(int fd, const void *buf, size_t count);
​
  • fd:文件描述符。
  • buf:要写入的数据缓冲区。
  • count:要写入的数据字节数。

示例代码:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);
    if (fd == -1) {
        perror("open");
        return 1;
    }
    const char *msg = "Hello, world!\n";
    ssize_t bytes_written = write(fd, msg, 14);
    if (bytes_written == -1) {
        perror("write");
        close(fd);
        return 1;
    }
    printf("Wrote %ld bytes\n", bytes_written);
    close(fd);
    return 0;
}
​

3.3 read系统调用

read系统调用用于从文件读取数据。其原型定义在 <unistd.h>头文件中:

#include <unistd.h>

ssize_t read(int fd, void *buf, size_t count);
​
  • fd:文件描述符。
  • buf:用于存储读取数据的缓冲区。
  • count:要读取的数据字节数。

示例代码:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd = open("example.txt", O_RDONLY);
    if (fd == -1) {
        perror("open");
        return 1;
    }
    char buf[128];
    ssize_t bytes_read = read(fd, buf, sizeof(buf) - 1);
    if (bytes_read == -1) {
        perror("read");
        close(fd);
        return 1;
    }
    buf[bytes_read] = '\0';
    printf("Read %ld bytes: %s\n", bytes_read, buf);
    close(fd);
    return 0;
}
​

3.4 close系统调用

close系统调用用于关闭文件描述符。其原型定义在 <unistd.h>头文件中:

#include <unistd.h>

int close(int fd);
​
  • fd:要关闭的文件描述符。

示例代码:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd = open("example.txt", O_WRONLY | O_CREAT, 0644);
    if (fd == -1) {
        perror("open");
        return 1;
    }
    printf("File opened with fd: %d\n", fd);
    if (close(fd) == -1) {
        perror("close");
        return 1;
    }
    printf("File closed\n");
    return 0;
}
​

四、理解“一切皆文件”

在Linux中,一切皆文件。这意味着所有的IO操作(包括文件、设备、网络通信等)都通过文件描述符进行。这种设计简化了系统调用的接口,使得程序可以用统一的方式处理不同类型的IO设备。

4.1 文件

常规文件通过文件描述符进行读写操作,如前文所述的 openreadwriteclose

4.2 设备

设备文件(如 /dev/null/dev/sda)也可以通过文件描述符操作。例如,读取系统内存信息:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd = open("/dev/mem", O_RDONLY);
    if (fd == -1) {
        perror("open");
        return 1;
    }
    // 读取内存数据的操作...
    close(fd);
    return 0;
}
​

4.3 网络

网络套接字也通过文件描述符操作。以下是一个简单的TCP客户端示例:

#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <unistd.h>
#include <stdio.h>
#include <string.h>

int main() {
    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if (sockfd == -1) {
        perror("socket");
        return 1;
    }

    struct sockaddr_in server_addr;
    memset(&server_addr, 0, sizeof(server_addr));
    server_addr.sin_family = AF_INET;
    server_addr.sin_port = htons(8080);
    inet_pton(AF_INET, "127.0.0.1", &server_addr.sin_addr);

    if (connect(sockfd, (struct sockaddr *)&server_addr, sizeof(server_addr)) == -1) {
        perror("connect");
        close(sockfd);
        return 1;
    }

    const char *msg = "Hello, server!";
    send(sockfd, msg, strlen(msg), 0);

    char buf[128];
    ssize_t bytes_received = recv(sockfd, buf, sizeof(buf) - 1, 0);
    if (bytes_received == -1) {
        perror("recv");
        close(sockfd);
        return 1;
    }
    buf[bytes_received] = '\0';
    printf("Received: %s\n", buf);

    close(sockfd);
    return 0;
}
​

五、总结

本文详细介绍了Linux中的进程IO与系统调用,包括 openwritereadclose函数及其用法,解释了文件描述符(fd)的概念,并深入探讨了Linux中的“一切皆文件”思想。这种设计极大地简化了系统编程,使得处理不同类型的IO设备变得更加一致和简单。通过本文的学习,您应该能够更好地理解和应用Linux中的进程IO操作,提高系统编程的效率和能力。

目录
相关文章
|
网络协议 Linux 调度
深入探索Linux操作系统的心脏:内核与系统调用####
本文旨在揭开Linux操作系统中最为核心的部分——内核与系统调用的神秘面纱,通过生动形象的语言和比喻,让读者仿佛踏上了一段奇妙的旅程,从宏观到微观,逐步深入了解这两个关键组件如何协同工作,支撑起整个操作系统的运行。不同于传统的技术解析,本文将以故事化的方式,带领读者领略Linux内核的精妙设计与系统调用的魅力所在,即便是对技术细节不甚了解的读者也能轻松享受这次知识之旅。 ####
|
缓存 算法 安全
深入理解Linux操作系统的心脏:内核与系统调用####
【10月更文挑战第20天】 本文将带你探索Linux操作系统的核心——其强大的内核和高效的系统调用机制。通过深入浅出的解释,我们将揭示这些技术是如何协同工作以支撑起整个系统的运行,同时也会触及一些常见的误解和背后的哲学思想。无论你是开发者、系统管理员还是普通用户,了解这些基础知识都将有助于你更好地利用Linux的强大功能。 ####
256 1
|
Docker 容器
14 response from daemon: open \\.\pipe\docker_engine_linux: The system cannot find the file speci
14 response from daemon: open \\.\pipe\docker_engine_linux: The system cannot find the file speci
499 1
|
存储 Linux 程序员
Linux中的主要系统调用
【9月更文挑战第11天】在Linux操作系统中,通过系统调用`fork`创建新进程,子进程继承父进程的数据结构与代码,但可通过`execve`执行不同程序。`fork`返回值区分父子进程,`waitpid`让父进程等待子进程结束。
|
Linux 开发者 Python
从Windows到Linux,Python系统调用如何让代码飞翔🚀
【9月更文挑战第10天】在编程领域,跨越不同操作系统的障碍是常见挑战。Python凭借其“编写一次,到处运行”的理念,显著简化了这一过程。通过os、subprocess、shutil等标准库模块,Python提供了统一的接口,自动处理底层差异,使代码在Windows和Linux上无缝运行。例如,`open`函数在不同系统中以相同方式操作文件,而`subprocess`模块则能一致地执行系统命令。此外,第三方库如psutil进一步增强了跨平台能力,使开发者能够轻松编写高效且易维护的代码。借助Python的强大系统调用功能,跨平台编程变得简单高效。
438 1
|
存储 Linux 缓存
linux中文件描述符fd和文件指针flip的理解
整理自:http://www.cnblogs.com/Jezze/archive/2011/12/23/2299861.html 简单归纳:fd(file descriptor)只是一个整数,在open时产生。
1674 0
|
7月前
|
Linux 应用服务中间件 Shell
二、Linux文本处理与文件操作核心命令
熟悉了Linux的基本“行走”后,就该拿起真正的“工具”干活了。用grep这个“放大镜”在文件里搜索内容,用find这个“探测器”在系统中寻找文件,再用tar把东西打包带走。最关键的是要学会使用管道符|,它像一条流水线,能把这些命令串联起来,让简单工具组合出强大的功能,比如 ps -ef | grep 'nginx' 就能快速找出nginx进程。
835 1
二、Linux文本处理与文件操作核心命令
|
7月前
|
Linux
linux命令—stat
`stat` 是 Linux 系统中用于查看文件或文件系统详细状态信息的命令。相比 `ls -l`,它提供更全面的信息,包括文件大小、权限、所有者、时间戳(最后访问、修改、状态变更时间)、inode 号、设备信息等。其常用选项包括 `-f` 查看文件系统状态、`-t` 以简洁格式输出、`-L` 跟踪符号链接,以及 `-c` 或 `--format` 自定义输出格式。通过这些选项,用户可以灵活获取所需信息,适用于系统调试、权限检查、磁盘管理等场景。
482 137
|
7月前
|
安全 Ubuntu Unix
一、初识 Linux 与基本命令
玩转Linux命令行,就像探索一座新城市。首先要熟悉它的“地图”,也就是/根目录下/etc(放配置)、/home(住家)这些核心区域。然后掌握几个“生存口令”:用ls看周围,cd去别处,mkdir建新房,cp/mv搬东西,再用cat或tail看文件内容。最后,别忘了随时按Tab键,它能帮你自动补全命令和路径,是提高效率的第一神器。
1299 58
|
6月前
|
存储 安全 Linux
Linux卡在emergency mode怎么办?xfs_repair 命令轻松解决
Linux虚拟机遇紧急模式?别慌!多因磁盘挂载失败。本文教你通过日志定位问题,用`xfs_repair`等工具修复文件系统,三步快速恢复。掌握查日志、修磁盘、验重启,轻松应对紧急模式,保障系统稳定运行。
1185 2

热门文章

最新文章