本文主要介绍一下可重入函数与线程安全的区别与联系,在此之前我们先来了解一些基本概念:什么是线程全函数,什么是可重入函数?

线程安全函数

  • 概念

线程安全的概念比较直观,一般来说,一个函数被称为线程安全的,当且仅当被多个并发线程反复调用时,它会一直产生正确的结果。

  • 确保线程安全

要确保函数线程安全,主要需要考虑的是线程之间的共享变量。属于同一线程的不同进程会共享进程内存空间中的全局区和堆,而私有的线程空间则主要包括栈和寄存器。因此,对于同一进程的不同线程来说,每个线程的局部变量都是私有的,而全局变量、局部静态变量、分配于堆的变量都是共享的。而对于这些共享变量进行访问时,如果要保证线程安全,则必须通过加锁的方式。

  • 线程不安全的后果

线程不安全可能导致的后果是显而易见的——共享变量的值由于不同线程的访问,可能发生不可预料的变化,进而导致程序的错误,甚至崩溃。

可重入函数

  • 概念

当捕捉到信号时,不论进程的主控制流程当前执行到哪,都会先跳到信号处理函数中执行,从信号处理函数返回后再继续执行主控制流程。信号处理函数是一个单独的控制流程,因为它和主控制流程是异步的,二者不存在调用和被调用的关系,并且使用不同的堆栈空间。引入了信号处理函数使得一个进程具有多个控制流程,如果这些控制流程访问相同的全局资源(全局变量、硬件资源等),就有可能出现冲突。
如下是不可重入的函数:

像上例这样,insert函数被不同的控制流程调用,有可能在第一次调用还没返回时就再次进入该函 数,这称为重入,insert函数访问一个全局链表有可能因为重入而造成错乱,像这样的函数称为 不可重入函数,反之,如果一个函数只访问自己的局部变量或参数,则称为可重入(Reentrant) 函数。

为了便于理解以上分析过程,在这里附上信号捕捉流程图:

可重入的概念基本没有比较正式的完整解释,这里有个大体的理解,所谓的“重入”,常见的情况是,程序执行到某个函数foo(),收到信号,于是暂停目前正在执行的函数,转到信号处理函数,而这个信号处理函数的执行过程中,又恰恰也会进到刚刚执行的函数foo(),这样便发生了所谓的重入。此时如果foo()能够正确的运行,而且处理完成后,之前暂停的foo()也能够正确运行,则说明它是可重入的。

Linux中可重入这个概念一般只有在signal的场景下有意义,叫async-signal-safe。很多线程安全的函数都是不可重入的,例如malloc。可重入函数一般也是线程安全的,当然据说是有反例的。

  • 确保可重入

要确保函数可重入,需要满足一下几个条件:

  1. 不在函数内部使用静态或全局数据
  2. 不返回静态或全局数据,所有数据都由函数的调用者提供
  3. 使用本地数据,或者通过制作全局数据的本地拷贝来保护全局数据
  4. 不调用不可重入函数
  5. 不使用用malloc或者new开辟出的空间
       其实总结就是,一个可重入函数内部使用的数据都应该来自于自身的栈空间,包括返回值也不应该是全局或者静态的;而正是因为其中的操作数据都来自于自身的栈空间,而每次调用函数会开辟不同的栈空间,因此二者互不影响。

  • 不可重入的后果

不可重入的后果主要体现在像信号处理函数这样需要重入的情况中。如果信号处理函数中使用了不可重入的函数,则可能导致程序的错误甚至崩溃。

可重入与线程安全

可重入与线程安全并不等同。一般来说,可重入的函数一定时线程安全的,但反过来说就不一定成立了,也就是说,可重入是比线程安全要要求严格的,这里我们可以用一张数学上的韦恩图来表示:

接下来我们来用例子来结合上图来进行进一步说明:

  • 如果一个函数中用到了全局或静态变量,那么它不是线程安全的,也不是可重入的;
  • 如果我们对它加以改进,在访问全局或者静态变量时使用互斥量或信号量等方式加锁,则可以使它变成线程安全的,但此时它仍然是不可重入的,因为通常加锁方式是针对不同线程的访问,而对同一线程可能出现问题;
  • 如果将函数中的全局或静态变量去掉,改成函数参数等其他形式,则有可能使函数变成既线程安全,又可重入。

比如:strtok函数是既不可重入的,也不是线程安全的;加锁的strtok不是可重入的,但线程安全;而strok_r既是可重入的,也是线程安全的。

最后总结一下可重入函数与线程安全的区别与联系:

  • 线程安全是在多个线程的情况下引发的,而可重入函数可以是在一个线程情况下而言的;

  • 线程安全不一定是可重入的,而可重入函数则一定是线程安全的;

  • 如果一个函数中有全局变量,那么这个函数既不是线程安全也不是可重入的;

  • 如果将对临界资源的访问加上锁,则这个函数是线程安全的,但如果重入函数的话若锁还未释放则会产生死锁,因此不是可重入的;

  • 如果一个函数当中的数据全是自身栈空间的,那么这个函数既是线程安全也是可重入的;

  • 线程安全函数能够使不同的线程访问同一块地址空间,而可重入函数要求不同的执行流对数据的操作互不影响使结果是相同的。


可重入函数与线程安全的区别与联系相关推荐

  1. 可重入函数 与线程安全的区别与联系

    线程安全:多个线程访问同一个区域的时候其最终结果是可预期的,并不会因为产生冲突或者异常中断再次恢复而使结果不可预期 1.重入:函数被不同的控制流程调用,有可能在第一次调用还没有返回的时候就再次进入该函 ...

  2. 线程安全与可重入函数的区别及联系

    一.线程安全 如果你的代码所在的进程中有多个线程在同时运行,而这些线程可能会同时运行这段代码.如果每次运行结果和单线程运行的结果是一样的,而且其他的变量的值也和预期的是一样的,就是线程安全的.  或者 ...

  3. 14.线程安全?线程不安全?可重入函数?不可重入函数?

    线程安全问题 基本定义 线程安全:简单来说线程安全就是多个线程并发执行同一段代码时,不会出现不同的结果,我们就可以说该线程是安全的: 线程不安全:如果多线程并发执行时会产生不同的结果,则该线程就是不安 ...

  4. 线程安全和可重入函数的联系与区别

    1.    线程安全: 线程安全是多线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程访问完,其他线程才可以使用.不会出现数据不一致或数据污染. 线程 ...

  5. 线程安全与可重入函数

    线程安全:一个函数被称为线程安全的(thread-safe),当且仅当被多个并发进程反复调用时,它会一直产生正确的结果.如果一个函数不是线程安全的,我们就说它是线程不安全的(thread-unsafe ...

  6. [Linux]线程安全和可重入函数

    线程安全:一个函数被称为线程安全的,当且仅当被多个并发进程反复调用时,它会一直产生正确的结果.如果一个函数不是线程安全的,我们就说它是线程不安全的. 重入:函数被不同的控制流程调用,有可能在第一次调用 ...

  7. 线程安全与可重入函数的区别与联系

    线程安全 线程安全是多个线程访问时,采用了加锁机制,当一个线程访问该类的某个数据时,进行保护,其他线程不能进行访问直到该线程读取结束并且释放了锁,其他线程才可使用,保证了数据的一致性. 与之对应的则是 ...

  8. 可重入性和线程安全性

    可重入性和线程安全性均与函数处理资源的方式有关. 但是,它们是不同的: 可重入函数既不会在连续调用中存储静态数据,也不会返回指向静态数据的指针. 对于这种类型的函数,调用方将提供函数所需的所有数据,如 ...

  9. Linux可重入函数

    1.可重入函数 可重入函数指的是可以被中断的函数.也就是说,可以在这个函数执行的任何时刻中断它,转入OS调度下去执行另外一段代码,而返回控制时不会出现什么错误:而不可重入的函数由于使用了一些系统资源, ...

最新文章

  1. OpenCV 【一】—— OpenCV中数组指针、图像分块计算、指针取像素值与MatToEigen方法,内存对齐
  2. Mysql8.0.16 only_full_group_by
  3. Hiredis库的简单使用
  4. 端口安全原理介绍及配置命令
  5. easyui accordion全部是关闭状态
  6. 操作系统原理: 计算机的体系结构、内存的层次结构 、地址的生成
  7. 协议编码分析 - ARP协议详解
  8. 李迟2021年7月知识总结
  9. AJAX vs FLEX:执行,传送,解析JSON,HTML,XML,AFM格式效率比较.
  10. 强化学习: Q-learning实例python实现
  11. 前端路由UmiJs快速上手
  12. Win10免费升级win11方法
  13. MessageBox.Show()的用法
  14. win32编程的经典书籍
  15. web兼容性面试题及答案
  16. MapObject 控件的使用之加入图层(作者/张松伟)
  17. 百度网盘链接 转存失败 解决方法
  18. Libra只是出了白皮书 中国央行的数字货币研究所已经做了落地试点
  19. 51单片机太阳能风能风光互补路灯控制器设计
  20. 第二届国信蓝点杯 c语言 本科组 赛题分析 第9题

热门文章

  1. 马尔科夫随机场简单理解
  2. c# IE浏览器清除缓存没用
  3. 邦纳传感器M18TUP8Q
  4. N1・N2听力单词 —— 交通、出行 / 家庭生活、人际关系
  5. Redis新版本发布,你还认为Redis是单线程?
  6. CSDN-markdown语法之如何使用LaTeX语法编写数学公式
  7. 用java求水仙花数,适合新手
  8. 知识图谱-知识抽取(三):非结构化数据【DeepDive:基于远程监督的“关系抽取”】【斯坦福开发的开源知识抽取系统,通过弱监督学习,从非结构化的文本中抽取结构化的关系数据 】
  9. 数控车椭圆编程实例带图_数控车床椭圆怎么编程
  10. 破解加密文档无法搜索复制问题、扫描文件无法复制的问题