导读

strerror是否线程安全了? 1

errno是否线程安全? 1

附1:strerror源码 2

附2:__strerror_r源码 2

strerror是否线程安全了?

答案是NO,但它有个线程安全的版本:strerror_r。借助Linux的man,即可看到详情:

#include <string.h>

char *strerror(int errnum);

int strerror_r(int errnum, char *buf, size_t buflen); /* GNU-specific */

那么,在多线程中使用strerror是否安全了?答案是不一定,一定情况下也是非常安全的。

不安全会造成内存违规访问吗?也就是会发生SIGSEGV吗?答案是NO,仍是内存安全的,但是可能会返回错乱的字符串。

那么,在多线程程序中,什么情况下使用strerror是绝对安全的了?如果参数errnum是一个已知的errno,则使用strerror是绝对安全的,也就是会返回期待的字符串,而不会出现乱码。

对比strerror源码,是因为strerror会在下述直接返回:

if (__builtin_expect (ret != NULL, 1))

return ret;

而这走的是_strerror_r中的分支:

return (char *) _(_sys_errlist_internal[errnum]);

errno是否线程安全?

答案是不一定,取决于编译宏:

#  if !defined _LIBC || defined _LIBC_REENTRANT

/* When using threads, errno is a per-thread value.  */

#   define errno (*__errno_location ())

#  endif

可以通过下段小代码,来确定默认时是否有定义,如果没有,则需要在编译时加上:

#include <stdio.h>

int main()

{

#ifdef _GNU_SOURCE

printf("_GNU_SOURCE defined\n");

#else

printf("_GNU_SOURCE not defined\n");

#endif

#ifdef _LIBC_REENTRANT

printf("_LIBC_REENTRANT defined\n");

#else

printf("_LIBC_REENTRANT not defined\n");

#endif

#ifdef _LIBC

printf("_LIBC defined\n");

#else

printf("_LIBC not defined\n");

#endif

return 0;

}

附1:strerror源码

// glibc-2.14\string\strerror.c

#include <errno.h>

/* Return a string describing the errno code in ERRNUM.

The storage is good only until the next call to strerror.

Writing to the storage causes undefined behavior.  */

libc_freeres_ptr (static char *buf);

char *

strerror (errnum)

int errnum;

{

char *ret = __strerror_r (errnum, NULL, 0);

int saved_errno;

if (__builtin_expect (ret != NULL, 1))

return ret;

saved_errno = errno;

if (buf == NULL)

buf = malloc (1024);

__set_errno (saved_errno);

if (buf == NULL)

return _("Unknown error");

return __strerror_r (errnum, buf, 1024);

}

附2:__strerror_r源码

// glibc-2.14\string\_strerror.c

/* Return a string describing the errno code in ERRNUM.  */

char *

__strerror_r (int errnum, char *buf, size_t buflen)

{

if (__builtin_expect (errnum <  || errnum >= _sys_nerr_internal

|| _sys_errlist_internal[errnum] == NULL, 0))

{

/* Buffer we use to print the number in.  For a maximum size for

`int' of 8 bytes we never need more than 20 digits.  */

char numbuf[21];

const char *unk = _("Unknown error ");

size_t unklen = strlen (unk);

char *p, *q;

bool negative = errnum < 0;

numbuf[20] = '\0';

p = _itoa_word (abs (errnum), &numbuf[20], 10, 0);

/* Now construct the result while taking care for the destination

buffer size.  */

q = __mempcpy (buf, unk, MIN (unklen, buflen));

if (negative && unklen < buflen)

{

*q++ = '-';

++unklen;

}

if (unklen < buflen)

memcpy (q, p, MIN ((size_t) (&numbuf[21] - p), buflen - unklen));

/* Terminate the string in any case.  */

if (buflen > 0)

buf[buflen - 1] = '\0';

return buf;

}

return (char *) _(_sys_errlist_internal[errnum]);

}

weak_alias (__strerror_r, strerror_r)

libc_hidden_def (__strerror_r)

strerror线程安全分析的更多相关文章

  1. Java线程问题分析定位

    Java线程问题分析定位 分析步骤: 1.使用top命令查看系统资源占用情况,发现Java进程占用大量CPU资源,PID为11572: 2.显示进程详细列表命令:ps -mp 11572 -o THR ...

  2. Android系统--输入系统(七)Reader_Dispatcher线程启动分析

    Android系统--输入系统(七)Reader_Dispatcher线程启动分析 1. Reader/Dispatcher的引入 对于输入系统来说,将会创建两个线程: Reader线程(读取事件) ...

  3. Android系统--输入系统(十三)Dispatcher线程情景分析_Reader线程传递事件

    Android系统--输入系统(十三)Dispatcher线程情景分析_Reader线程传递事件 1. 输入按键 我们知道Android系统的按键分为三类:(1)Global Key;(2)Syste ...

  4. Android系统--输入系统(十四)Dispatcher线程情景分析_dispatch前处理

    Android系统--输入系统(十四)Dispatcher线程情景分析_dispatch前处理 1. 回顾 我们知道Android输入系统是Reader线程通过驱动程序得到上报的输入事件,还要经过处理 ...

  5. Spring中获取request的几种方法,及其线程安全性分析

    前言 本文将介绍在Spring MVC开发的web系统中,获取request对象的几种方法,并讨论其线程安全性. 原创不易,如果觉得文章对你有帮助,欢迎点赞.评论.文章有疏漏之处,欢迎批评指正. 欢迎 ...

  6. 【转】java线上程序排错经验2 - 线程堆栈分析

    前言 在线上的程序中,我们可能经常会碰到程序卡死或者执行很慢的情况,这时候我们希望知道是代码哪里的问题,我们或许迫切希望得到代码运行到哪里了,是哪一步很慢,是否是进入了死循环,或者是否哪一段代码有问题 ...

  7. [No000016E]Spring 中获取 request 的几种方法,及其线程安全性分析

    前言 本文将介绍在Spring MVC开发的web系统中,获取request对象的几种方法,并讨论其线程安全性. 原创不易,如果觉得文章对你有帮助,欢迎点赞.评论.文章有疏漏之处,欢迎批评指正. 欢迎 ...

  8. java线程dump分析工具

    jstack和线程dump分析  java程序性能分析之thread dump和heap dump 一.[内存dump] jmap –dump:live,format=b,file=heap.bin ...

  9. Spring中获取request的几种方法,及其线程安全性分析(山东数漫江湖)

    前言 本文将介绍在Spring MVC开发的web系统中,获取request对象的几种方法,并讨论其线程安全性. 原创不易,如果觉得文章对你有帮助,欢迎点赞.评论.文章有疏漏之处,欢迎批评指正. 欢迎 ...

随机推荐

  1. ThinkPHP 配置详解

      3.0 ThinkPHP配置详解 3.1 入口文件的配置 一般不建议在入口文件做过多的配置,但可以重新定义一些系统常量,以下简单介绍几个常用的系统常量. 1.APP_PATH 默认情况下,框架的项 ...

  2. CentOS7 安装Chrome

    1. 下载Chrome浏览器的rpm包 https://www.chrome64bit.com/index.php/google-chrome-64-bit-for-linux 2. 安装Chrome ...

  3. 分析java类的初始化契机

    分析java类的静态成员变量初始化先于非静态成员变量   依上图中当class字节码文件被jvm虚拟机加载到内存中依次经过 连接 验证:对字节码进行验证 准备:给静态变量分配内存并赋予变量类型各自的默 ...

  4. Python Twisted系列教程19:改变之前的想法

    作者:dave@http://krondo.com/i-thought-i-wanted-it-but-i-changed-my-mind/  译者: Cheng Luo 你可以从”第一部分 Twis ...

  5. Word域介绍文章

    https://www.cnblogs.com/ahuo/archive/2007/05/04/735520.html pageref 书签名 :返回书签所在页码 styleref 标题 1 在当前位 ...

  6. Numpy的ndarry

    Numpy的ndarry:一种多维数组对象 Numpy最重要的一个特点就是其N维数组对象(即ndarry),该对象是一个快速而灵活的大数据集容器.你可以利用这种数组对整块数据执行一些数学运算,其语法跟 ...

  7. python 协程 gevent 简单测试

    串行测试 from gevent import monkey; monkey.patch_all()#有IO才做时需要这一句 import gevent import requests,time st ...

  8. JAVA_01

    Java局部变量 局部变量声明在方法.构造方法或者语句块中: 局部变量在方法.构造方法.或者语句块被执行的时候创建,当它们执行完成后,变量将会被销毁: 访问修饰符不能用于局部变量: 局部变量只在声明它 ...

  9. LNMP 参数调优 ( 无注释 )

    简介: PHP FastCGI 优点 1.PHP 脚本运行速度更快.PHP 解释程序被载入内存而不用每次需要时从存储器读取,极大的提升了依靠脚本运行站点的性能. 2.需要使用的系统资源更少.由于服务器 ...

  10. Asp .net core api+Entity Framework core 实现数据CRUD数据库中(附Git地址)

    最近在学dotNetCore 所以尝试了一下api 这个功能 不多说了大致实现如下 1.用vs2017建立一个Asp.net  Core Web 应用程序 在弹出的对话框中选择 Web API 项目名 ...