可重入和不可重入

个可重入的函数简单来说就是可以被中断的函数,也就是说,可以在这个函数执行的任何时刻中断它,转入OS调度下去执行另外一段代码,而返回控制时不会出现什么错误;而不可重入的函数由于使用了一些系统资源,比如全局变量区,中断向量表等,所以它如果被中断的话,可能会出现问题,这类函数是不能运行在多任务环境下也不应该被中断的。
  也可以这样理解,重入即表示重复进入,首先它意味着这个函数可以被中断,其次意味着它除了使用自己栈上的变量以外不依赖于任何环境(包括static),这样的函数就是purecode(纯代码)可重入,可以允许有该函数的多个副本在运行,由于它们使用的是分离的栈,所以不会互相干扰。如果确实需要访问全局变量(包括static),一定要注意实施互斥手段。可重入函数在并行运行环境中非常重要,但是一般要为访问全局变量付出一些性能代价。
  编写可重入函数时,若使用全局变量,则应通过关中断、信号量(即P、V操作)等手段对其加以保护,但是往往有些系统函数是不可重入的。而且对于信号处理函数,在信号处理函数汇总一定要使用可重入函数,这样当信号到来引起中断,完成信号处理函数之后回到原函数不会破坏原函数中的数据。如果在信号处理函数中没有使用可重入函数那么可能引起以下的情况:

 ************************************

重点:

     * 不含全局变量和静态变量是可重入函数的一个要素
     * 在信号捕捉函数里应使用可重入函数

****************************************

 

所以在信号捕捉函数里禁止调用不可重入函数,例如:strtok就是一个不可重入函数,因为strtok内部维护了一个内部静态指针,保存上一次切割到的位置,如果信号的捕捉函数中也去调用strtok函数,则会造成切割字符串混乱,应用strtok_r版本,r表示可重入。

1、strtok函数

函数原型:char * strtok (char *str, const char * delimiters);
参数:str,待分割的字符串(c-string);delimiters,分割符字符串。
该函数用来将字符串分割成一个个片段。参数str指向欲分割的字符串,参数delimiters则为分割字符串中包含的所有字符。当strtok()在参数s的字符串中发现参数delimiters中包涵的分割字符时,则会将该字符改为\0 字符。在第一次调用时,strtok()必需给予参数s字符串,往后的调用则将参数s设置成NULL。每次调用成功则返回指向被分割出片段的指针。
需要注意的是,使用该函数进行字符串分割时,会破坏被分解字符串的完整,调用前和调用后的s已经不一样了。第一次分割之后,原字符串str是分割完成之后的第一个字符串,剩余的字符串存储在一个静态变量中,因此多线程同时访问该静态变量时,则会出现错误。
strtok例子
 1 #include <stdio.h>  
 2 #include <stdlib.h>  
 3 #include <string.h>  
 4   
 5 int main()  
 6 {  
 7     char str[]="ab,cd,ef";  
 8     char *ptr;  
 9     printf("before strtok:  str=%s\n",str);  
10     printf("begin:\n");  
11     ptr = strtok(str, ",");  
12     while(ptr != NULL){  
13         printf("str=%s\n",str);  
14         printf("ptr=%s\n",ptr);  
15         ptr = strtok(NULL, ",");  
16     }  
17     system("pause");  
18     return 0;  
19 }  
输出结果如下:
before strtok:  str=ab,cd,ef
begin:
str=ab
ptr=ab
str=ab
ptr=cd
str=ab
ptr=ef

2、strtok_s函数

strtok_s是windows下的一个分割字符串安全函数,其函数原型如下:
char *strtok_s( char *strToken, const char *strDelimit, char **buf);
这个函数将剩余的字符串存储在buf变量中,而不是静态变量中,从而保证了安全性。

3strtok_r函数

strtok_s函数是linux下分割字符串的安全函数,函数声明如下:
char *strtok_r(char *str, const char *delim, char **saveptr);
该函数也会破坏带分解字符串的完整性,但是其将剩余的字符串保存在saveptr变量中,保证了安全性。
例子:
 1 #include <stdio.h>  
 2 #include <stdlib.h>  
 3 #include <string.h>  
 4   
 5 int main()  
 6 {  
 7     char str[]="ab,cd,ef";  
 8     char *ptr;  
 9     char *p;  
10     printf("before strtok:  str=%s\n",str);  
11     printf("begin:\n");  
12     ptr = strtok_r(str, ",", &p);  
13     while(ptr != NULL){  
14         printf("str=%s\n",str);  
15         printf("ptr=%s\n",ptr);  
16         ptr = strtok_r(NULL, ",", &p);  
17     }  
18     return 0;  
19 }  

 

posted @ 2017-02-17 01:10  柳下_MBX  阅读(700)  评论(0编辑  收藏  举报