#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <pthread.h>
#include <unistd.h>
pthread_mutex_t lk = PTHREAD_MUTEX_INITIALIZER;
pthread_t thread_id2;
void work1(void *arg)
{
pthread_mutex_lock(&lk);
puts("thread 1 working");
pthread_mutex_unlock(&lk);
sleep(3);
pthread_mutex_lock(&lk);
pthread_cancel(thread_id2);
pthread_mutex_unlock(&lk);
int seconds = 5;
while(seconds--)
{
pthread_mutex_lock(&lk);
puts("thread2 was canceled, thread 1 continue working;");
pthread_mutex_unlock(&lk);
sleep(1);
}
}
void cleanupfuc(void *arg)
{
pthread_mutex_lock(&lk);
puts("thread 2 is cleanup");
pthread_mutex_unlock(&lk);
}
void work2(void *arg)
{
pthread_cleanup_push(cleanupfuc, NULL);
//pthread_cleanup_pop(1);
while(1)
{
pthread_testcancel();
sleep(1);
pthread_mutex_lock(&lk);
puts("thread 2 working");
pthread_mutex_unlock(&lk);
}
pthread_cleanup_pop(0);
}
int main()
{
pthread_t thread_id1;
pthread_create(&thread_id1, NULL, (void *)work1, NULL);
pthread_create(&thread_id2, NULL, (void *)work2, NULL);
pthread_join(thread_id1, NULL);
pthread_join(thread_id2, NULL);
return 0;
}
PS:需要注意的是在线程被取消(关于线程的取消,参考上一篇博文)的时候,无论 pthread_cleanup_pop的参数是否为0,都会触发清理函数。
那么为什么还要设置一个参数呢?此参数是用来强制执行前面注册的清理函数的,我们可以看到第42行被注释了。
实际上如果把第42行取消注释的话(取消注释的时候,需要注释第51行的pthread_cleanup_pop调用,因为linux下这两个函数的实现为宏,必须成对使用。)
假如此时传入的execute为0,那么清理函数不会被触发。因为检查点在它之后。假如此时execute为1,那么将提前触发清理函数。
pthread_cleanup_push每次压入一个清理函数,pthread_cleanup_pop每次将其弹出。所以必须成对使用这两个函数。
某种程度上来说,清理函数的调用是一次性的。