阅读 943

初学者必看的Linux errno 错误码机制

不同于Java的异常处理机制, 当你使用C更多的接触到是基于错误码的异常机制, 简单来说就是当调用的函数发生异常时, 程序不会跳转到一个统一处理异常的地方, 取而代之的是返回一个整型错误码。

可能会有小伙伴有疑问了, 以打开文件为例该函数定义如下所示

int open(const char *pathname, int flags); 复制代码

如果打开文件成功, open函数会返回一个文件描述符(该值大于0), 如果失败则返回-1。对于开发者来说, 只知道文件打开失败了, 而却不知道具体原因, 实际上的原因可能是多种多样的, 如:

  • 文件不存在

  • 当前进程没有该文件的读写权限

那么该如何知道具体错误呢? 这就需要借助系统为我们提供的errno机制了。

errno

errno是一个定义在errno.h头文件的全局整型变量,表示当前发生的最后一个错误, 只需在代码中引用errno.h这个头文件边可以获取到这个值。

如以下demo所示, 我们尝试打开一个不存在的文件:

#include <stdio.h> #include <errno.h> #include <stdlib.h> #include <fcntl.h> int main() {     int fd;     fd = open("/test.log", O_RDONLY);     if (fd == -1) {         printf("open failed, errno: %d\n", errno);     }     return 0; } 复制代码

运行该程序, 输出如下所示:

open failed, errno: 2 复制代码

可以看出,此时错误码为2, 怎么知道这个错误码代表什么意思呢?

我们可以通过以下方式获取错误信息

moreutils

  1. 安装moreutils

apt install moreutils 复制代码

  1. 运行errno 错误码查看具体的错误信息

image.png 可以验证, 错误码2表示当前文件或目录不存在,与我们的预期一致

perror

使用定义在stdio.h中的perror函数可以直接在标准输出上打印错误信息

该函数定义如下所示, 我们可以在错误信息前附加自己定义的错误信息。

void perror(const char *s); 复制代码

Demo:

int main() {     int fd;     fd = open("/test.log", O_RDONLY);     if (fd == -1) {         perror("open failed");     }     return 0; } 复制代码

image.png

strerr

如果我们只需要获取错误码对应的文本以记录日志, 可以使用strerr函数, 该函数定义在string.h

Demo:

int main() {     int fd;     fd = open("/test.log", O_RDONLY);     if (fd == -1) {       char* err_msg = strerror(errno);       printf("%s\n", err_msg);     }     return 0; } 复制代码

输出
image.png

线程安全的吗?

相信对于并发问题比较敏感的同学已经意识到了一个问题:这errno是一个整型的全局变量, 那如果多个线程同时执行系统调用, 并且都因为不同的原因失败了, 会不会导致其他线程的错误信息全部被最后一个产生错误的线程给覆盖掉了? 以及会不会有线程安全问题呢?

实际上errno被定义为了线程局部变量, 概念同Java中的ThreadLocal, 即每个线程都会有自己的errno变量, 不同线程之间不会互相影响。

总结

  • errno 代表着上一次调用函数发生的具体错误信息

  • 对于系统调用我们可以通过perror打印错误信息, strerror获取对应的错误信息文本

  • errno是线程局部变量, 每个线程都有自己的errno, 因此是线程安全的

注意事项

  • 不要直接使用errno判断函数调用是否成功, 而应该先判断函数的返回结果,再使用errno获取正确的错误信息。(errno保存的是最后一次调用失败的错误码)


作者:kovogo
链接:https://juejin.cn/post/7028151650300723230


文章分类
代码人生
版权声明:本站是系统测试站点,无实际运营。本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 XXXXXXo@163.com 举报,一经查实,本站将立刻删除。
相关推荐