【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);
​
AI 代码解读
  • 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;
}
​
AI 代码解读

3.2 write系统调用

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

#include <unistd.h>

ssize_t write(int fd, const void *buf, size_t count);
​
AI 代码解读
  • 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;
}
​
AI 代码解读

3.3 read系统调用

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

#include <unistd.h>

ssize_t read(int fd, void *buf, size_t count);
​
AI 代码解读
  • 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;
}
​
AI 代码解读

3.4 close系统调用

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

#include <unistd.h>

int close(int fd);
​
AI 代码解读
  • 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;
}
​
AI 代码解读

四、理解“一切皆文件”

在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;
}
​
AI 代码解读

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;
}
​
AI 代码解读

五、总结

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

目录
打赏
0
31
34
4
446
分享
相关文章
【JavaEE】——文件IO(万字长文)
文件路径,文本文件,二进制文件,File类,文件流,字节流(InputStream,OutputStream)字符流(Reader,Writer)
Maven clean 提示文件 java.io.IOException
在使用Maven进行项目打包时,遇到了`Failed to delete`错误,尝试手动删除目标文件也失败,提示`java.io.IOException`。经过分析,发现问题是由于`sys-info.log`文件被其他进程占用。解决方法是关闭IDEA和相关Java进程,清理隐藏的Java进程后重新尝试Maven clean操作。最终问题得以解决。总结:遇到此类问题时,可以通过任务管理器清理相关进程或重启电脑来解决。
【文件IO】实现:查找文件并删除、文件复制、递归遍历目录查找文件
【文件IO】实现:查找文件并删除、文件复制、递归遍历目录查找文件
85 2
【文件IO】文件内容操作
【文件IO】文件内容操作
99 2
|
5月前
|
【文件IO】文件系统操作
【文件IO】文件系统操作
77 1
Java 流(Stream)、文件(File)和IO的区别
Java中的流(Stream)、文件(File)和输入/输出(I/O)是处理数据的关键概念。`File`类用于基本文件操作,如创建、删除和检查文件;流则提供了数据读写的抽象机制,适用于文件、内存和网络等多种数据源;I/O涵盖更广泛的输入输出操作,包括文件I/O、网络通信等,并支持异常处理和缓冲等功能。实际开发中,这三者常结合使用,以实现高效的数据处理。例如,`File`用于管理文件路径,`Stream`用于读写数据,I/O则处理复杂的输入输出需求。
367 12
【Java】文件IO
【Java】文件IO
64 0
C语言 文件IO (系统调用)
本文介绍了Linux系统调用中的文件I/O操作,包括文件描述符、`open`、`read`、`write`、`lseek`、`close`、`dup`、`dup2`等函数,以及如何获取文件属性信息(`stat`)、用户信息(`getpwuid`)和组信息(`getgrgid`)。此外还介绍了目录操作函数如`opendir`、`readdir`、`rewinddir`和`closedir`,并提供了相关示例代码。系统调用直接与内核交互,没有缓冲机制,效率相对较低,但实时性更高。
linux中在进程之间传递文件描述符的实现方式
linux中在进程之间传递文件描述符的实现方式
|
15天前
|
Linux 进程前台后台切换与作业控制
进程前台/后台切换及作业控制简介: 在 Shell 中,启动的程序默认为前台进程,会占用终端直到执行完毕。例如,执行 `./shella.sh` 时,终端会被占用。为避免不便,可将命令放到后台运行,如 `./shella.sh &`,此时终端命令行立即返回,可继续输入其他命令。 常用作业控制命令: - `fg %1`:将后台作业切换到前台。 - `Ctrl + Z`:暂停前台作业并放到后台。 - `bg %1`:让暂停的后台作业继续执行。 - `kill %1`:终止后台作业。 优先级调整:
32 5

热门文章

最新文章

AI助理

你好,我是AI助理

可以解答问题、推荐解决方案等