标准C库函数之perror()、strerror(),以及他们之间各种的优缺点(打印错误信息)

标准C库函数之perror()、strerror(),以及他们之间各种的优缺点(打印错误信息)

前言
如果,想要深入的学习标准C库中perror、strerror函数,还是需要去自己阅读Linux系统中的帮助文档。

具体输入命令:

man 3 perror
man 3 strerror
1
2
即可查阅到完整的资料信息。

perror函数
perror函数是C语言标准库中的一个函数,用于在发生系统调用错误时输出有关错误信息的描述。该函数定义在stdio.h头文件中。perror根据传入的参数(通常是一个字符串)生成一个描述性的错误消息,然后将这个消息输出到标准错误流(stderr)。

函数原型如下:

#include <stdio.h> //使用此函数需引入此头文件

void perror(const char *str);
1
2
3
参数:

str:一个指向常量字符的指针,用于提供上下文信息。在输出错误消息时,这个字符串将作为错误消息的前缀(也就是说,你可以手动的去描述这个错误是怎么发生的)。
perror的工作原理:当一个系统调用或库函数发生错误时,通常会将全局变量errno设置为一个特定的错误码。perror函数读取errno的值,并根据这个值生成相应的错误描述。然后,将错误描述与传入的字符串参数拼接,并输出到标准错误流。

使用示例:

#include <stdio.h>
#include <stdlib.h>

int main() {
FILE *file_ptr;

file_ptr = fopen("non_existent_file.txt", "r");
if (file_ptr == NULL) {perror("Error opening file");return 1;
}// ... 其他操作 ...fclose(file_ptr);
return 0;

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
在上面的示例中,当试图打开一个不存在的文件时,fopen函数会返回NULL,并将errno设置为一个特定的错误码。

调用perror函数输出类似于以下内容的错误消息:

Error opening file: No such file or directory
1
这里的"Error opening file"是传入的字符串参数,而"No such file or directory"是由perror根据errno值生成的描述性错误信息。

strerror 函数
strerror函数是标准C库中的一种函数,它用于生成描述错误代码的字符串。这个函数在string.h头文件中定义。

这个函数的原型是:

#include <string.h> //使用此函数需引入此头文件

char *strerror(int errnum);
1
2
3
参数:

该函数接受一个整数errnum作为参数,这个参数通常是一个错误代码,比如你可能从某些函数(如open或read)中得到这样的错误代码。
返回值:

函数会返回一个指向描述该错误的字符串的指针。这个字符串可以用来打印错误消息,使得用户能够理解发生了什么问题。
例如,你可以这样使用strerror:

#include <stdio.h>
#include <string.h>
#include <errno.h>

int main() {
FILE *file = fopen(“non_existent_file.txt”, “r”);
if (file == NULL) {
printf(“打开文件出错: %s\n”, strerror(errno));
return 1;
}

// 做一些处理
fclose(file);
return 0;

}
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
在这个例子中,如果文件不存在,fopen会返回NULL,并将errno设置为表示错误的值。然后,我们可以使用strerror来获取并打印描述这个错误的消息。

需要注意的是,strerror返回的字符串是静态分配的,不需要(也不应该)被手动释放。此外,因为strerror返回的是静态数据,所以它不是线程安全的。在多线程环境中,你可能需要使用strerror_r,这个函数有一个额外的参数用于存储结果,可以避免竞态条件。

另外,也应该注意strerror可能不会知道所有可能的错误代码。如果你传递一个它不认识的错误代码,它可能会返回一个消息说这个错误未知。

strerror()函数与perror()函数各种的优缺点
strerror

优点:

灵活性:strerror返回一个字符串,这使得你可以自由地使用这个字符串。例如,你可以将其存储,发送给用户,或者记录到日志中。

自定义错误消息:可以通过与其他字符串拼接,创建定制的错误消息。

缺点:

需要手动处理错误消息:在打印错误消息时,需要手动处理错误消息的格式化和输出。

非线程安全:strerror在多线程环境中可能会产生问题,因为它返回静态内存中的字符串。如果你需要线程安全,需要使用strerror_r版本。

perror

优点:

简单易用:perror在打印错误消息时无需处理错误消息的格式化和输出,使得错误处理代码更加简洁。

自动生成错误消息:它自动地在你提供的自定义消息后添加一个描述错误的消息。

缺点:

不够灵活:perror总是将错误消息输出到标准错误输出。如果你需要将错误消息发送到其他地方,你就无法使用perror。

缺乏自定义:虽然你可以提供自定义消息,但perror控制的消息格式有限。如果你需要更特殊的格式,可能需要使用strerror。

总的来说,选择使用哪一个函数取决于你的需求。如果你需要处理错误消息,例如将其发送到特定的日志系统,或者创建定制的错误消息,那么strerror可能是更好的选择。如果你只需要快速地向用户报告错误,并且默认的格式足够好,那么perror可能是更好的选择。

strerror_r 函数
strerror_r函数是标准C库中的另一种处理错误的函数,它与strerror函数类似,但具有线程安全性。

在标准C库中,strerror_r函数有两个不同的实现:POSIX和GNU。它们具有不同的原型和行为,我将分别介绍它们。
POSIX版本的strerror_r

POSIX版本的strerror_r的原型如下:

int strerror_r(int errnum, char *buf, size_t buflen);
1
这个函数接受三个参数:

errnum:错误码。
buf:用于存储错误消息的缓冲区。
buflen:缓冲区的大小。
返回值

函数的返回值是一个整数,表示操作的成功与否。如果返回值为0,表示成功获取错误消息;如果返回值为正数,表示错误码无效,无法获取错误消息;如果返回值为负数,表示发生了错误。

需要注意的是,POSIX版本的strerror_r函数的行为和strerror函数略有不同。它不返回一个静态字符串指针,而是将错误消息复制到提供的缓冲区中。因此,这个函数是线程安全的,因为它使用了用户提供的缓冲区。

GNU版本的strerror_r

GNU版本的strerror_r的原型如下:

char *strerror_r(int errnum, char *buf, size_t buflen);
1
这个函数接受三个参数:

errnum:错误码。
buf:用于存储错误消息的缓冲区。
buflen:缓冲区的大小。
返回值

函数的返回值是一个指向错误消息字符串的指针。与strerror函数不同,GNU版本的strerror_r并不返回静态分配的字符串,而是将错误消息复制到提供的缓冲区中,并返回指向缓冲区的指针。

这个版本的strerror_r在多线程环境中是安全的,因为它将错误消息存储在用户提供的缓冲区中,而不是使用静态内存。

需要注意的是,GNU版本的strerror_r与POSIX版本的返回类型和行为不同。在GNU版本中,返回值始终是指向错误消息字符串的指针,不论操作是否成功。

使用哪个版本取决于在编译时使用的特性测试宏。如果定义了 _POSIX_C_SOURCE,则使用 XSI-compliant version,否则使用 GNU-specific version。这也是为什么 strerror_r 函数在不同的系统和设置中可能有不同的行为。

总结
在我们的讨论中,我们涉及了以下内容:

strerror函数:它是标准C库中的函数,用于生成描述错误代码的字符串。它接受一个错误码作为参数,并返回一个描述错误的字符串。这个字符串可以用于打印错误消息或其他需要错误描述的场景。

perror函数:它也是标准C库中的函数,用于报告错误消息。它会打印你提供的自定义消息,后跟一个冒号、一个空格,然后是描述当前错误的消息。它的优点是简单易用,适用于快速报告错误的场景。

strerror_r函数:它是标准C库中的函数,用于获取描述错误代码的字符串,同时具备线程安全性。它有两个不同的实现:POSIX和GNU。POSIX版本将错误消息复制到用户提供的缓冲区中,并返回一个表示操作成功与否的整数。而GNU版本将错误消息复制到用户提供的缓冲区中,并返回指向缓冲区的指针。

总体而言,strerror函数和strerror_r函数用于获取描述错误的字符串,而perror函数用于报告错误消息。选择使用哪个函数取决于你的需求和编程环境,例如需要灵活性、自定义错误消息还是线程安全性。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/725002.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

网络编程作业day6

数据库操作的增、删、改完成 #include <myhead.h>//查询的回调函数 int callback(void* data,int count,char** argv, char** columnName) {//count是字段数//argv是字段内容//columnName是字段名称for(int i0;i<count;i) {printf("%s%s\n", columnName[…

2024全国水科技大会暨新能源及电子行业废水论坛(十一)

一、会议背景 为深入学习贯彻《中共中央、国务院关于全面推进美丽中国建设的意见》&#xff0c;全面贯彻实施《固体废物污染环境防治法》、《“十四五”全国城市基础设施建设规划》&#xff0c;推进我国污泥处理工程建设&#xff0c;提高处理产物资源化利用水平&#xff0c;促进…

堆排序实现

在了解堆的一些性质后&#xff0c;我们可以根据这些性质来实现一个较优的算法&#xff0c;也就是堆排序。 堆排序&#xff0c;顾名思义就是排序&#xff0c;其运用到堆的各种性质。首先我们要创建一个堆&#xff0c;在原有数组上进行向上或者向下调整使其变成一个堆。那么这时…

typedef的详细理解

目录 1.typedef对整型指针进行重命名 2.typedef对数组指针进行重命名 3.typedef对函数指针进行重命名 总结&#xff1a;所有重新命名的函数名必须写在*号旁边。 1.typedef对整型指针进行重命名 代码如下&#xff1a; typedef int* pint; int main() {int* p NULL;pint p…

Python内置模块

目录 什么是模块 模块分类 通过模块创建者分类 系统内置模块 第三方模块 在线安装 离线安装 模块导入 math和random模块介绍 math模块 random模块 什么是模块 在我们编写程序时&#xff0c;需要导入包。例如随机数的产生&#xff0c;需要import random。import XXX&…

计算机基础专升本笔记-汇总笔记(三)常用文件格式

计算机基础专升本笔记-汇总笔记&#xff08;三&#xff09;常用文件格式 一、文本文件格式 扩展名&#xff08;后缀名&#xff09;文件类型docxWord2010文档dotxWord2010模版xlsxExcel2010文档xltxExcel2010模版pptxPower Point 演示文稿potxPower Point 2010模版ppsxPower P…

XGboost的整理

XGboost&#xff08;extreme gradient boosting&#xff09;:高效实现了GBDT算法并进行了算法和工程上的许多改进。 XGboost的思路&#xff1a; 目标&#xff1a;建立k个回归树&#xff0c;使得树群的预测尽量接近真实值&#xff08;准确率&#xff09;而且有尽量大的泛化能力…

Java项目:39 springboot008房屋租赁系统

作者主页&#xff1a;舒克日记 简介&#xff1a;Java领域优质创作者、Java项目、学习资料、技术互助 文中获取源码 项目介绍 房屋租赁系统的设计基于现有的网络平台&#xff0c;主要有管理员、房东、用户三个角色 管理员可以在线查看系统已有的人中心、房屋类型管理、房屋信息…

在 Android 中使用 Kotlin 调用动态库

在 Android 开发中&#xff0c;有时候需要使用原生代码来执行一些特定的任务&#xff0c;比如高性能计算、访问硬件或者使用现有的 C/C 库。通常&#xff0c;这样的原生代码会被编译成动态库&#xff08;.so 文件&#xff09;&#xff0c;然后在应用中被调用。 本篇博客将介绍…

创建java项目

文章目录 一、安装idea二、创建一个java项目1.设置名称 项目路径 安装JDK&#xff08;自己选择一个位置存放JDK&#xff09;2.创建完成之后 在src文件夹下创package包 再在包下创建Class类3. hellojava类创建完成设置打印语句 输出HelloJava 一、安装idea 官网地址 二、创建一…

不买后悔的阿里云服务器租用价格表_优惠活动整理_2024新版

2024阿里云服务器优惠活动政策整理&#xff0c;阿里云99计划ECS云服务器2核2G3M带宽99元一年、2核4G5M优惠价格199元一年&#xff0c;轻量应用服务器2核2G3M服务器61元一年、2核4G4M带宽165元1年&#xff0c;云服务器4核16G10M带宽26元1个月、149元半年&#xff0c;云服务器8核…

面试经典150题(105-107)

leetcode 150道题 计划花两个月时候刷完之未完成后转&#xff0c;今天&#xff08;第2天&#xff09;完成了3道(105-107)150 105.&#xff08;191. 位1的个数&#xff09;题目描述&#xff1a; 编写一个函数&#xff0c;输入是一个无符号整数&#xff08;以二进制串的形式&am…

simplex算法的代码实现

def pivot(N, B, A, b, c, v, l, e):N对应非基本元变量下标,B对应基本元变量下标,A对应非基本元在约束方程组中的系数相反数形成的矩阵,b对应约束条件中小于等于号右边的数值集合,c对应目标函数中变量系数形成的集合,v对应当前目标函数的取值,l对应转出变量下标在B中的位置,e…

使用Http请求下载文件带来的问题

java.io.IOException: Broken pipe 当使用http请求的方式将文件作为响应内容给浏览器&#xff0c;这个时候如果浏览器未开启自动下载(浏览器可能会终止这个tcp连接)&#xff0c;文件不会下载成功&#xff0c;但是这个时候请求已经到达服务器&#xff0c;如果这个时候&#xff0…

Dynamo初学常识梳理(四)——Revit图元

希望想学 Dynamo 的小伙伴坚持住&#xff0c;每天积累一点点知识&#xff0c;Dynamo 你很快就能上手的&#xff01;Dynamo 并不是你想的那样难学哦&#xff01; 今天要讲的是如何从 Dynamo 中获取 Revit 的图元&#xff0c;这些节点很常用&#xff0c;不需要全背下来&#xff0…

指针运算笔试题解析

题目1&#xff1a; int main() { int a[5] { 1, 2, 3, 4, 5 }; int* ptr (int*)(&a 1); printf("%d %d", *(a 1), *(ptr - 1)); return 0; } ptr中存放了整个数组的地址&#xff0c;ptr是int*类型&#xff0c;&a1跳到5的地址后又被强制类…

Freesia Spring配置文件与其他文件

common-api模块【application.yml文件】 project:version: 来自顶级pom.xml的properties配置 --- spring:profiles:# 根据当前活跃环境指定对应的 application文件active: profiles.active --- spring:datasource:driver-class-name: com.p6spy.engine.spy.P6SpyDriverurl: jd…

算法DFS 复习

思路&#xff1a;for 代表的是每一位的纵向&#xff0c;数字变化&#xff0c;dfs 代表的是横向的&#xff0c;位置变化。vis 来做到每个枚举的数不重复&#xff0c;并且要在搜索前记录&#xff0c;搜索后还原。模拟该样例 dfs3 的时候是输出&#xff0c;dfs0&#xff0c;1&…

『大模型笔记』Ollama ModelFile(模型文件)

Ollama ModelFile(模型文件) 文章目录 一. Ollama 模型文件1.1. 格式1.2. 示例1.2.1. 基本`Modelfile`1.2.1. `模型文件`位于 [ollama.com/library](https://ollama.com/library)1.3. 说明1.3.1. From(必填)1.3.1.1. 从 llama2 构建1.3.1.2. 从 `bin` 文件构建1.3.2. PARAMETER…

【DAY05 软考中级备考笔记】线性表,栈和队列,串数组矩阵和广义表

线性表&#xff0c;栈和队列&#xff0c;串数组矩阵和广义表 2月28日 – 天气&#xff1a;阴转晴 时隔好几天没有学习了&#xff0c;今天补上。明天发工资&#xff0c;开心&#x1f604; 1. 线性表 1.1 线性表的结构 首先线性表的结构分为物理结构和逻辑结构 物理结构按照实…