linux应用:errno、perror、open、fopen

devtools/2025/3/5 12:16:03/

errno

errno 是一个全局变量,定义在 头文件中。当系统调用(如 open、read、write 等)或库函数执行失败时,会将一个错误码赋值给 errno。不同的错误码代表不同的错误类型,通过检查 errno 的值,可以判断具体发生了什么错误。

#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <cerrno>int main() {// 尝试打开一个不存在的文件int fd = open("nonexistent_file.txt", O_RDONLY);if (fd == -1) {// 检查 errno 的值if (errno == ENOENT) {std::cerr << "The file does not exist. errno value: " << errno << std::endl;} else {std::cerr << "An unknown error occurred. errno value: " << errno << std::endl;}}return 0;
}

代码中使用 open 函数尝试打开一个不存在的文件。由于文件不存在,open 函数会执行失败并返回 -1。
当 open 函数返回 -1 时,通过检查 errno 的值来判断具体的错误类型。ENOENT 是一个预定义的错误码,表示文件或目录不存在。如果 errno 等于 ENOENT,则输出相应的错误信息。

perror

perror 是一个函数,定义在 头文件中。它会根据 errno 的当前值,输出一条对应的错误信息。perror 函数接受一个字符串作为参数,该字符串会被作为错误信息的前缀输出。

#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <cstdio>int main() {// 尝试打开一个不存在的文件int fd = open("nonexistent_file.txt", O_RDONLY);if (fd == -1) {// 使用 perror 输出错误信息perror("Failed to open the file");}return 0;
}

同样,代码中使用 open 函数尝试打开一个不存在的文件,由于文件不存在,open 函数执行失败并返回 -1。
当 open 函数返回 -1 时,调用 perror 函数并传入一个字符串 “Failed to open the file” 作为前缀。perror 函数会根据 errno 的当前值,输出一条包含前缀和具体错误信息的错误消息,例如:Failed to open the file: No such file or directory。

#include <iostream>
#include <fcntl.h>
#include <unistd.h>
#include <cerrno>
#include <cstdio>int main() {// 尝试打开一个不存在的文件int fd = open("nonexistent_file.txt", O_RDONLY);if (fd == -1) {// 先使用 perror 输出错误信息perror("Open file error");// 再根据 errno 进行具体处理switch (errno) {case ENOENT:std::cerr << "The file does not exist." << std::endl;break;case EACCES:std::cerr << "Permission denied." << std::endl;break;default:std::cerr << "An unknown error occurred." << std::endl;break;}}return 0;
}

errno 和 perror 的使用。当 open 函数执行失败时,首先调用 perror 函数输出一条包含前缀和通用错误信息的错误消息。
然后,通过 switch 语句根据 errno 的具体值进行更细致的错误处理,输出不同的错误提示信息,方便开发者进行调试和问题定位。

文件IO和标准IO的区别

文件 I/O(File I/O)和标准 I/O(Standard I/O)是在编程中进行输入输出操作的两种不同方式,它们在多个方面存在区别,下面为你详细介绍:

1. 接口层面

文件 I/O
系统调用接口:文件 I/O 使用的是操作系统提供的系统调用函数,这些函数直接与操作系统内核进行交互。在 Linux 系统中,常见的文件 I/O 函数有 open、read、write、lseek 和 close 等。这些函数是操作系统内核的一部分,不同的操作系统可能会有不同的实现细节。
低级别操作:由于直接与内核交互,文件 I/O 提供了更底层的操作方式,允许程序员对文件的读写位置、读写权限等进行更精确的控制。
标准 I/O
库函数接口:标准 I/O 是 C 标准库提供的一组函数,如 fopen、fread、fwrite、fseek 和 fclose 等。这些函数是在文件 I/O 的基础上进行了封装,提供了更高级、更方便的接口。
跨平台性:标准 I/O 函数是 C 标准库的一部分,具有很好的跨平台性,在不同的操作系统上都能保持一致的使用方式。

2. 缓冲机制

文件 I/O
无缓冲或自定义缓冲:文件 I/O 默认情况下是无缓冲的,即每次调用 read 或 write 函数都会直接进行系统调用,与磁盘进行数据交互。不过,程序员可以自己实现缓冲机制来提高效率,例如使用 read 函数一次性读取较大的数据块到缓冲区,然后再进行处理。
适合实时性要求高的场景:由于无缓冲,文件 I/O 能够立即将数据写入磁盘或从磁盘读取数据,适合对实时性要求较高的场景,如日志记录、设备驱动等。
标准 I/O
自动缓冲:标准 I/O 提供了自动缓冲机制,分为全缓冲、行缓冲和无缓冲三种类型。全缓冲是指当缓冲区满时才进行实际的 I/O 操作;行缓冲是指当遇到换行符时才进行 I/O 操作;无缓冲则是每次操作都立即进行 I/O。
提高效率:自动缓冲机制减少了系统调用的次数,提高了 I/O 操作的效率。例如,在向文件写入大量数据时,标准 I/O 会先将数据写入缓冲区,当缓冲区满时再一次性将数据写入磁盘,减少了磁盘 I/O 的次数。

3. 文件定位

文件 I/O
使用 lseek 函数:文件 I/O 通过 lseek 函数来定位文件的读写位置。lseek 函数可以将文件指针移动到指定的偏移量处,支持相对当前位置、文件开头和文件末尾三种偏移方式。
灵活性高:lseek 函数提供了较高的灵活性,程序员可以根据需要精确地控制文件指针的位置,实现随机读写操作。
标准 I/O
使用 fseek 函数:标准 I/O 使用 fseek 函数来定位文件的读写位置。fseek 函数的功能与 lseek 类似,但它是基于标准 I/O 流的,使用起来更加方便。
与缓冲机制结合:由于标准 I/O 有缓冲机制,fseek 函数在移动文件指针时会处理缓冲区的内容,确保数据的一致性。

4. 错误处理

文件 I/O
使用 errno:文件 I/O 函数在执行失败时会设置全局变量 errno,通过检查 errno 的值可以判断具体的错误类型。同时,还可以使用 perror 函数输出更详细的错误信息。
底层错误信息:errno 提供的错误信息比较底层,与操作系统的具体实现相关,对于一些复杂的错误,需要对操作系统有一定的了解才能准确判断。
标准 I/O
返回值和 ferror 函数:标准 I/O 函数通过返回值来表示操作是否成功,例如 fread 和 fwrite 函数返回实际读写的元素个数,如果返回值与预期不符,则表示操作失败。此外,还可以使用 ferror 函数检查流是否发生错误。
高级错误处理:标准 I/O 提供的错误处理方式相对更高级,更易于理解和使用,适合初学者。

5. 适用场景

文件 I/O
系统编程:在进行系统编程、设备驱动开发等需要直接与操作系统内核交互的场景中,文件 I/O 是首选。例如,开发一个磁盘文件系统的工具,需要精确控制文件的读写和定位,使用文件 I/O 可以更好地满足需求。
实时性要求高的场景:对于实时性要求较高的应用,如日志记录、网络编程等,文件 I/O 的无缓冲特性可以确保数据及时写入磁盘或发送到网络。
标准 I/O
通用文件处理:在一般的文件处理场景中,如文本文件的读写、数据的存储和读取等,标准 I/O 的自动缓冲机制和高级接口可以提高开发效率,减少代码复杂度。
跨平台开发:由于标准 I/O 具有良好的跨平台性,在需要编写跨操作系统的程序时,使用标准 I/O 可以确保代码的可移植性。
示例代码对比

文件 I/O 示例

#include <iostream>
#include <fcntl.h>
#include <unistd.h>int main() {const char* filename = "test.txt";const char* data = "Hello, File I/O!";// 打开文件int fd = open(filename, O_CREAT | O_WRONLY, 0666);if (fd == -1) {std::cerr << "Failed to open the file." << std::endl;return 1;}// 写入数据ssize_t bytesWritten = write(fd, data, sizeof(data));if (bytesWritten == -1) {std::cerr << "Failed to write to the file." << std::endl;close(fd);return 1;}// 关闭文件close(fd);return 0;
}

标准 I/O 示例

#include <iostream>
#include <cstdio>int main() {const char* filename = "test.txt";const char* data = "Hello, Standard I/O!";// 打开文件FILE* file = fopen(filename, "w");if (file == nullptr) {std::cerr << "Failed to open the file." << std::endl;return 1;}// 写入数据size_t bytesWritten = fwrite(data, sizeof(char), sizeof(data), file);if (bytesWritten != sizeof(data)) {std::cerr << "Failed to write to the file." << std::endl;fclose(file);return 1;}// 关闭文件fclose(file);return 0;
}

http://www.ppmy.cn/devtools/164742.html

相关文章

批量设置 Word 样式,如字体信息、段落距离、行距、页边距等信息

在 Word 文档中&#xff0c;我们可以做各种样式的处理。比如设置 Word 文档的字体样式、设置 Word 文档的段落样式以及设置 Word 文档的页面样式。我们通常可以在 Office 中完成这些操作&#xff0c;相信绝大部分场景我们也是这样完成的。但是如果我们手上有 1000 个 Word 文档…

C#进阶入门

当你对 C# 有了基础的了解后&#xff0c;想要进一步提升自己的技能&#xff0c;可以从以下几个方面进行进阶学习&#xff1a; 1. 深入理解面向对象编程 继承与多态 继承允许一个类继承另一个类的属性和方法&#xff0c;而多态则允许你以统一的方式处理不同类型的对象。 // …

力扣HOT100之哈希:1. 两数之和

这道题之前刷代码随想录的时候已经刷过好几遍了&#xff0c;看到就直接秒了。这道题主要是通过unordered_map<int, int>来建立哈希表&#xff0c;其中键用来保存向量中的元素&#xff0c;而对应的值则为元素的下标。遍历整个向量&#xff0c;当遍历到nums[i]时&#xff0…

C++蓝桥杯基础篇(七)

片头 嗨~小伙伴们&#xff0c;大家好&#xff01;今天我们来一起学习蓝桥杯基础篇&#xff08;七&#xff09;&#xff0c;学习相关字符串的知识&#xff0c;准备好了吗&#xff1f;咱们开始咯&#xff01; 一、字符与整数的联系——ASCII码 每个常用字符都对应一个-128~127的…

Rust~String、str、str、String、Box<str> 或 Box<str>

Rust语言圣经中定义 str Rust 语言类型大致分为两种&#xff1a;基本类型和标准库类型&#xff0c;前者由语言特性直接提供&#xff0c;后者在标准库中定义 str 是唯一定义在 Rust 语言特性中的字符串&#xff0c;但也是几乎不会用到的字符串类型 str 字符串是 DST 动态大小…

2025-03-04 学习记录--C/C++-C语言 判断是否是素数

合抱之木&#xff0c;生于毫末&#xff1b;九层之台&#xff0c;起于累土&#xff1b;千里之行&#xff0c;始于足下。&#x1f4aa;&#x1f3fb; C语言 判断是否是素数 一、代码 ⭐️ #include <stdio.h> #include <stdbool.h> // 使用 bool 类型// 判断是否是…

Versal - XRT(CPP) 2024.1

目录 1.简介 2. XRT 2.1 XRT vs OpenCL 2.2 Takeways 2.3 XRT C APIs 2.4 Device and XCLBIN 2.5 Buffers 2.5.1 Buffer 创建 2.5.1.1 普通 Buffer 2.5.1.2 特殊 Buffer 2.5.1.3 用户指针 Buffer 2.5.2 Data Transfer 2.5.2.1 read/write API 2.5.2.2 map API 2…

请你说一下你对服务降级的理解

服务降级是指在服务出现故障或资源不足的情况下&#xff0c;通过牺牲部分功能或性能&#xff0c;以确保核心功能或服务的可用性。其主要目的是在系统面临压力或故障时&#xff0c;仍然能够提供基本的服务&#xff0c;而不是完全失败。 服务降级的核心思想 保证核心功能&#x…