使用PTHREAD_CANCEL_DEFERRED方式并不能完全避免这个问题,因为无法保证在获取临界资源后(比如lock操作)不会进行可以作为取消点的操作(如进行sleep),此时主线程如果对该线程发送cancel信号,线程将会在不释放锁的情况下直接结束运行,即还是会出现在释放资源前线程就退出的问题。
为了避免上述情况,不仅需要设置可取消类型,还需要设置可取消状态。将获取临界资源-释放临界资源之间的代码块都设置成PTHREAD_CANCEL_DISABLE状态,其余的代码块都设置成PTHREAD_CANCEL_ENABLE状态,确保线程在安全的地方退出。如果在可以安全退出的代码块不存在取消点系统调用,可以调用pthread_testcancel函数自己添加取消点。
伪代码描述如下:
void* subThread(void*)
{
pthread_setcancelstate(PTHREAD_CANCEL_ENABLE,&oldCancleState);
…;//不存在获取临界资源操作,可以安全退出的代码块
pthread_testcancel();//如果可以安全退出的代码块不存在取消点操作,可以自己添加pthread_testcancel调用,线程执行到这个调用就会退出
/*还有一种方法,在可以安全退出的代码块,我们将线程的可取消类型设置成PTHREAD_CANCEL_ ASYNCHRONOUS,这样即使没有取消点也可以马上退出*/
pthread_setcancelstate(PTHREAD_CANCEL_DISABLE,&oldCancleState);
/*存在获取-释放临界资源操作,如果在lock和unlock之间的运行收到cancel信号,且可取消状态为enable,则锁永远无法被释放*/
Lock();
…;
Unlock();
}
void* mainThread(void*)
{
pthread_cancel(subThread);//给subThread发送退出信号
pthread_join(subThread,null);//进入休眠,直到subThread退出成功
}
无论使用哪种方式,核心点就是要保证线程退出的时候不会获取了某些临界资源而无法释放
POSIX.1定义的取消点见下:


注意:当主线程调用pthread_cancel接口后,只是将取消请求发送给指定线程,
对接口的成功调用不能保证指定线程已经退出,需要调用pthread_join等待指定线程完全退出,再进行相关资源的释放。
以上就是小编为大家带来的Linux线程退出方式总结(推荐)全部内容了,希望大家多多支持易采站长站~








