C语言函数重入
C语言函数重入
- 可重入函数:可以被中断的函数,即这个函数执行时,可以中断其执行,可以由一个或多个任务并发使用,而不比担心数据错误。
- 不可重入函数(不安全函数)
- 不能运行在多任务环境下,除非能保证互斥(使用信号量/代码的关键部分禁用中断)
- 是由于使用了未受保护的系统资源,如全局变量区,中断向量表等。
可重入函数:
- 没有静态数据结构
- 不返回指向静态数据的指针
- 所有函数数据由函数的调用者提供
- 使用auto变量,或通过全局变量的拷贝来保护全局变量
- 若必须访问全局变量,则利用互斥信号保护
- 不调用不可重入函数
不可重入函数有:
如果一个函数在重入条件下使用了未受保护的共享的资源,那么它是不可重入的。
- 函数中使用了静态变量,无论是全局静态变量还是局部静态变量。
- 函数返回静态变量
- 函数中调用了不可重入函数
- 函数体内使用了静态的数据结构
- 函数体内调用了malloc()或者free()函数
- 函数体内调用了其他标准I/O函数
- 函数是singleton中的成员函数,而且使用了不使用线程独立存储的成员变量
不可重入函数改写成可重入函数:
1、不使用全局变量
2、在和硬件发生交互时,可能会发生中断时,先关闭中断(有些系列叫做“进入/退出核心”或者用OS_ENTER_KERNAL/OS_EXIT_KERNAL来描述,这是临界区保护)
3、不调用不可重入函数
4、谨慎使用堆栈,最好在使用前OS_ENTER_KERNAL
中断是嵌入式系统中重要的组成部分,这导致了很多编译开发商提供一种扩展—让标准C支持中断。中断服务子程序ISR:
__interrupt double compute_area (double radius) { double area = PI * radius * radius; printf("\nArea = %f", area); return area; }
错误:
1、ISR没有返回值
如果它有返回值,返回给谁,某种中断源产生中断,系统使用ISR进行处理。ISR是链接在某一个中断源上的,而中断源的产生是随机的,所以ISR并没有一个固定的调用者,也没有固定的返回地址,所以返回值没有用。
2、ISR不能传递参数
3、在许多编译器/处理器中,浮点数是不可重入的。有些不允许在ISR中做浮点运算。
4、ISR应该短、有效率的,做浮点数不明智。
5、printf是不可重入函数。
解释:
1、浮点数:一般浮点运算都是由专门的硬件来完成,举个例子假设有个硬件寄存器名字叫做FLOAT,用来计算和存放浮点数的中间运算结果。
假设有这么个函数
void fun() { //...这个函数对FLOAT寄存器进行操作 }
假如第一次执行,有个对浮点数操作运算的结果临时存在FLOAT寄存器中,而就在这时被中断了,而中断函数或者另一个进程也调用fun函数,这时第二次调用的fun函数在执行的过程中就会破坏第一次FLOAT寄存器中的结果,这样当返回到第一次fun函数的时候,结果就不正确了。
2、printf函数
引用全局变量stdout,这是标准输入输出流的一个对象
malloc --------全局内存分配表
free --------全局内存分配表
在unix里面通常都有加上_r后缀的同名可重入函数版本。
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 从 HTTP 原因短语缺失研究 HTTP/2 和 HTTP/3 的设计差异
· AI与.NET技术实操系列:向量存储与相似性搜索在 .NET 中的实现
· 基于Microsoft.Extensions.AI核心库实现RAG应用
· 阿里巴巴 QwQ-32B真的超越了 DeepSeek R-1吗?
· 【译】Visual Studio 中新的强大生产力特性
· 10年+ .NET Coder 心语 ── 封装的思维:从隐藏、稳定开始理解其本质意义
· 【设计模式】告别冗长if-else语句:使用策略模式优化代码结构
· 字符编码:从基础到乱码解决