写在前面
1. 本文内容对应《 UNIX 环境高级编程》 ( 第 2 版 ) 》第 1 章。
2. 主要介绍 errno 变量的使用,包括如何打印出错消息,以及多线程下的 errno 变量。
3. 希望本文对您有所帮助,也欢迎您给我提意见和建议。
errno
变量
当函数出错时,常常返回一个负值 (-1) ,而且整型变量 errno 通常被设置为含有附加信息的一个值。例如, open 函数如果成功执行则返回一个非负的文件描述符,如果出错则返回 -1 。在 open 出错时,有大约 15 种不同的 errno 值(如文件不存在,权限问题等)。变量 errno 定义在文件 <errno.h> 中。在 linux 下,可以赋予它的各种常量在 <asm-generic/errno.h> 和 <asm-generic/errno-base.h> 文件中定义,共 131 个,这些常量都以 E 开头,具体含义可以参考注释,也可以在 man 中查询( man errno )。
使用 errno 变量,应当记住两条规则:
l 如果没有出错,则其值不会被一个函数改变。因此,仅当函数的返回值指明出错时,才检查 errno 。
l 任一函数不会将 errno 的值设置为 0 ,在 <errno.h> 中定义的所有常量都不为 0 。
打印出错消息
C 标准定义了两个函数用于打印出错信息。
#include <string.h> char *strerror(int errnum);
#include <stdio.h> void perror(const char *msg); |
strerror 函数将 errnum (通常就是 errno )映射为一个出错信息字符串,并且返回此字符串的指针。 perror 函数基于 errno 的当前值,在标准出错上产生一条出错消息,然后返回。它首先输出由 msg 指向的字符串,然后是一个冒号,一个空格,接着是对应于 errno 值的出错消息,最后是一个换行符。使用范例如下:
#include <errno.h> #include <stdio.h> #include <string.h>
int main(int argc, char *argv[]) { fprintf(stderr, "EACCES: %s/n", strerror(EACCES)); errno = ENOENT; perror(argv[0]); exit(0); } |
其输出为:
pydeng@pydeng-laptop:~/apue.2e/mytest$ ./a.out EACCES: Permission denied ./a.out: No such file or directory |
多线程下的
errno在支持线程的环境中,多个线程共享进程的地址空间,但是每个线程都有属于它自己的局部 errno ,以避免一个线程干扰另一个线程。测试程序如下:
#include <stdio.h> #include <errno.h> #include <pthread.h>
pthread_t ntid;
void *thr_fn1(void *arg) { errno = 100; printf("thread %u: errno=%d/n", (unsigned int)pthread_self(), errno); }
void *thr_fn2(void *arg) { printf("thread %u: errno=%d/n", (unsigned int)pthread_self(), errno); }
int main() { int err;
errno = 131; pthread_create(&ntid, NULL, thr_fn1, NULL); pthread_create(&ntid, NULL, thr_fn2, NULL); exit(0); } |
由于 pthread 库不是 Linux 系统默认的库,连接时需要使用库 libpthread.a ,所以用 gcc 编译时要加 -lpthread 参数,如 gcc thread_errno.c -lpthread 。运行结果如下, errno 在主线程和两个新创建的线程之间保持独立。
pydeng@pydeng-laptop:~/apue.2e/mytest$ ./a.out thread 3085015952: errno=100 thread 3076623248: errno=0 |