多线程程序设计之---线程安全与可重入函数

为了写一个稳定的多线程程序,必须遵守线程安全,但不一定遵守可重入。
    安全是指多个线程调用同一个函数,如果是线程安全的,那么每次的结果都是正确。
    可重入函数是指函数内部没有使用共享变量。
    可重入函数是线程安全函数的一个真子集。
    也就是说如果函数是可重入的,就可以保证它是线程安全的,当然有些不可重入的函数也是线程安全的,比如系统库函数实现的都是线程安全的。
    系统的设计中,经常会出现多个任务调用同一个函数的情况。如果这个函数不幸被设计成为不可重入的函数的话,那么不同任务调用这个函数时可能修改其他任务调用这个函数的数据,从而导致不可预料的后果。那么什么是可重入函数呢?

    所谓可重入函数是指一个可以被多个任务调用的过程,任务在调用时不必担心数据是否会出错。不可重入函数在实时系统设计中被视为不安全函数。

int global=0;
int foo1()
{
    return global++;
}

如果多个线程同时调用这个函数,每个线程返回的结果是什么?

很明显这个结果很难回答。因为你不知道他们这些线程是怎么工作的,谁先谁后都不知道,有的程序在线程的调度上只是简单的时间片轮转的策略,按找创建线程的顺序应该可以得到正确答案,但是如果调度策略对修改了呢,比如是实时调度。

很明显一个可重入的函数的很必要的。

你或许想难道这个可重入函数不能使用全局变量了吗,你真正需要的或许只是线程安全,没有必要可重入,那么你可以这样改

int global=0;
int foo1()
{
    int local=global;
    return local++;
}
 

这就是为什么你或许看到很多程序都要在开始用一个局部变量保存一个全局变量的原因。

第二种情况 果函数中使用了静态变量或者返回静态变量或者静态数据结构,静态变量包括全局静态变量或者局部静态变量,函数也有是不可重入的,例如

int foo2()
{
    static int a=0;
    return a++;
}

要使它变得线程安全,其实很简单,在使用它之前,保存它的值。或者另一种情况这些变量都是只读的。

第三种情况 数中使用了malloc或者free函数,天呢,线程没有动态内存分配?不是的,malloc和free是不可重入的,但是是线程安全的,也就是开始所说的概念,所以你大可以继续malloc和free。

第四种情况 函数体内调用了其他标准I/O函数

第五种情况 函数是singleton中的成员函数而且使用了不使用线程独立存储的成员变量 。

第六种情况 函数调用了不可冲入函数

下面一个函数就是线程安全但却不可重入的

static int *sharedID;
int* threadsafe_getID( char* name )
{
int *unsharedID;
P( &mutex );
sharedID = notThreadsafe_getID( name );
unsharedID = sharedID;
V( & mutex);
return unsharedID;
}

上面的函数通过P、V操作封装得到一个线程安全函数,却不是可重入函数。

最后给出可重入函数编写规范得链接

<script type=text/javascript charset=utf-8 src="http://static.bshare.cn/b/buttonLite.js#style=-1&uuid=&pophcol=3&lang=zh"></script> <script type=text/javascript charset=utf-8 src="http://static.bshare.cn/b/bshareC0.js"></script>
阅读(557) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~
评论热议
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值