一个可重入的函数简单来说就是可以被中断的函数,也就是说,可以在这个函数执行的任何时刻中断它,转入OS调度下去执行另外一段代码,而返回控制时不会出现什么错误;而不可重入的函数由于使用了一些系统资源,比如全局变量区,中断向量表等,所以它如果被中断的话,可能会出现问题,这类函数是不能运行在多任务环境下也不应该被中断的。
也可以这样理解,重入即表示重复进入,首先它意味着这个函数可以被中断,其次意味着它除了使用自己栈上的变量以外不依赖于任何环境(包括static),这样的函数就是purecode(纯代码)可重入,可以允许有该函数的多个副本在运行,由于它们使用的是分离的栈,所以不会互相干扰。如果确实需要访问全局变量(包括static),一定要注意实施互斥手段。可重入函数在并行运行环境中非常重要,但是一般要为访问全局变量付出一些性能代价。
编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即P、V操作)等手段对其加以保护,但是往往有些系统函数是不可重入的。而且对于信号处理函数,在信号处理函数汇总一定要使用可重入函数,这样当信号到来引起中断,完成信号处理函数之后回到原函数不会破坏原函数中的数据。如果在信号处理函数中没有使用可重入函数那么可能引起以下的情况:
所以在信号捕捉函数里禁止调用不可重入函数,例如:strtok就是一个不可重入函数,因为strtok内部维护了一个内部静态指针,保存上一次切割到的位置,如果信号的捕捉函数中也去调用strtok函数,则会造成切割字符串混乱,应用strtok_r版本,r表示可重入。
strtok_r函数的举例:
// char *strtok_r(char *str, const char *delim, char **saveptr);
// 在str中以delim为分割符吧str中的delim转为\0,并吧剩下的字符串保存到saveptr中
// 返回被分割的字符串
#include <stdio.h>
#include <string.h>
int main(void)
{
char buf[] = "hello world itcast xwp";
char *save = buf, *p;
while((p = strtok_r(save, " ", &save)) != NULL)
printf("%s\n", p);
return 0;
}
注:man 7 signal可查看重入函数相关介绍