strerror() 和perror()函数

在linux编程中,strerror()是个好东东,因为一个孤零零的errno看不出个所以然,然而strerror()返回的错误描述已经给我们解决问题提供了80%的成功率。但从安全性的角度来讲,strerror_r是更好的选择,因为:
#include <string.h>
char *strerror(int errnum);
int strerror_r(int errnum, char *buf, size_t n);
 说明,对于函数strerror_r,第一个参数errnum是错误代码,第二个参数buf是用户提供的存储错误描述的缓存,第三个参数n是缓存的大小。
1,在系统编程中错误通常通过函数返回值来表示,并通过特殊变量errno来描述。

errno这个全局变量在<errno.h>头文件中声明如下:extern int errno;

errno是一个由POSIX和ISO C标准定义的符号,看(用)起来就好像是一个整形变量。当系统调用或库函数发生错误的时候,比如以只读方式打开一个不存在的文件时,它的值将会被改变,根据errno值的不同,我们就可以知道自己的程序发生了什么错误,然后进行相应的处理。

为什么,要强调errno看起来好像是一个整形变量呢?因为有的标准(如ISO C)只规定了errno的作用,而没有规定它的实现方式,它可能被定义成一个变量,也有可能被定义成一个宏,这个具体要看编译器自己的实现。早些时候,POSIX.1曾把errno定义成extern int errno这种形式,但现在这种方式比较少见了。因为以这种形式来实现errno,在多线程环境下errno变量是被多个线程共享的,这样可能线程A发生某些错误改变了errno的值,线程B虽然没有发生任何错误,但是当它检测errno的值的时候,线程B会以为自己发生了错误。所以现在errno在Linux中被实现成extern int * __errno_location(void): #define errno (*__errno_location()),这样每个线程都有自己的errno,不会再发生混乱了。

关于errno有三点需要特别注意:

1、如果系统调用或库函数正确执行的话,errno的值是不会被清零(置0,注意这里是不会被清零,不是不会被改变)的,假若执行函数A的时候发生了错误errno被改变,接下来直接执行函数B,如果函数B正确执行的话,errno还保留函数A发生错误时被设置的值。所以,在利用errno之前,最好先对函数的返回值进行判断,看是否发生了错误,返回值错误再利用errno判断时哪里发生了错误。所以如果一个函数无法从返回值上判断正误,而只能通过errno来判断出错,那你在调用它之前必须手动将errno清零!

2、系统调用或库函数正确执行,并不保证errno的值不会被改变!

3、任何错误号(即发生错误时errno的取值)都是非0的。

综上所述,当需要用errno来判断函数是否正确执行的时候,最好先将errno清零,函数执行结束时,通过其返回值判断函数是否正确执行,若没有正确执行,再根据errno判断时哪里发生了错误。
也可以用这个函数
#include <stdio.h>
函数原型为void perror(const char *s);
时间: 2024-10-24 20:45:29

strerror() 和perror()函数的相关文章

Linux/Unix C编程之的perror函数,strerror函数,errno

#include <stdio.h> // void perror(const char *msg); #include <string.h> // char *strerror(int errnum); #include <errno.h> //errno ? errno 是错误代码,在 errno.h头文件中: perror是错误输出函数,输出格式为:msg:errno对应的错误信息(加上一个换行符): strerror?是通过参数 errnum (就是errno)

文件IO之——阻塞和非阻塞及perror函数

读常规文件是不会阻塞的,不管读多少字节,read一定会在有限的时间内返回.从终端设备或网络读则不一定,如果从终端输入的数据没有换行符,调用read读终端设备就会阻塞,如果网络上没有接收到数据包,调用read从网络读就会阻塞,至于会阻塞多长时间也是不确定的,如果一直没有数据到达就一直阻塞在那里.同样,写常规文件是不会阻塞的,而向终端设备或网络写则不一定.现在明确一下阻塞(Block)这个概念.当进程调用一个阻塞的系统函数时,该进程被置于睡眠(Sleep)状态,这时内核调度其它进程运行,直到该进程等

perror 函数

perror编辑 perror( ) 用来将上一个函数发生错误的原因输出到标准设备(stderr).参数 s 所指的字符串会先打印出,后面再加上错误原因字符串.此错误原因依照全局变量errno 的值来决定要输出的字符串. 在库函数中有个errno变量,每个errno值对应着以字符串表示的错误类型.当你调用"某些"函数出错时,该函数已经重新设置了errno的值.perror函数只是将你输入的一些信息和现在的errno所对应的错误一起输出. 中文名 perror 头文件 #include&

C语言 perror函数

C语言 perror函数 perror 头文件: #include<stdio.h> 函数原型: void perror(char * string) 函数说明: 输出将最近一次发生错误的原因输出到标准错误(stderr),其中string会先输出到标准错误中,后面再加上错误原因字符串,其错误原因会依照全局变量errno的值 **errno全局变量说明:**在库函数中有个errno变量,每个errno值对应着以字符串表示的错误类型.当你调用"某些"函数出错时,该函数已经重新

c中perror函数

写代码这么久,竟然很少用到perror函数,忘记了其强大的功能. 所在头文件: #include<stdio.h> 函数定义: void perror(const char *str); 函数功能: 将上一个函数发生错误的原因输出到标准设备(stderr).参数str为所先打印的字符串,后面跟错误原因字符串.错误原因字符串取决于全局变量errno的值. 原文地址:https://www.cnblogs.com/rohens-hbg/p/9067979.html

APUE_1.7ErrorHandling Figure1.8Demonstrate strerror and perror

man 3 errno /* * 1.8DemonstrateStrerrorAndPerror.cpp * * Created on: Feb 11, 2015 * Author: sunyj */ #include "../apuesunyj.h" int main(int argc, char *argv[]) { fprintf(stderr, "EACCES: %s\n", strerror(EACCES)); errno = ENOENT; perror

信号处理篇alarm ferror kill mkfifo pause pclose perror pipe popen sigaction sigaddset sigdelset sigemptyset signal sleep strerror

alarm(设置信号传送闹钟) 相关函数 signal,sleep 表头文件 #include<unistd.h> 定义函数 unsigned int alarm(unsigned int seconds); 函数说明 alarm()用来设置信号SIGALRM在经过参数seconds指定的秒数后传送给目前的进程.如果参数seconds 为0,则之前设置的闹钟会被取消,并将剩下的时间返回. 返回值 返回之前闹钟的剩余秒数,如果之前未设闹钟则返回0. 范例 #include<unistd.h

perror 与 strerror

errno.stderr.perror函数和strerror函数 errno表示错误代码. 记录系统的最后一次错误代码.代码是一个int型的值,在errno.h中定义.系统每一次出错都会对应一个出错代码,例如12表示“Cannot allocate memory". stderr 是linux(unix)标准出错输出.linux中的一个进程启动时,都会打开三个文件:标准输入.标准输出和标准出错处理.通常这三个文件都与终端联 系.这三个文件分别对应文件描述符0.1.2.系队统自定义了三个文件指针s

linux编程中接收主函数返回值以及错误码提示

程序A创建子进程,并调用进程B,根据不调用的不同情况,最后显示结果不同. #include <stdio.h> #include <unistd.h> #include <sys/wait.h> #include <sys/types.h> #include <errno.h> int main() { pid_t pid, rpid; int stat; if ((pid = fork()) < 0) { perror("for