• (转载)Linux 线程取消(Pthread_cancel)


    线程取消(pthread_cancel)

    基本概念
    pthread_cancel调用并不等待线程终止,它只提出请求。线程在取消请求(pthread_cancel)发出后会继续运行,
    直到到达某个取消点(CancellationPoint)。取消点是线程检查是否被取消并按照请求进行动作的一个位置.


    与线程取消相关的pthread函数
    int pthread_cancel(pthread_t thread)
    发送终止信号给thread线程,如果成功则返回0,否则为非0值。发送成功并不意味着thread会终止。

    int pthread_setcancelstate(int state,   int *oldstate)   
    设置本线程对Cancel信号的反应,state有两种值:PTHREAD_CANCEL_ENABLE(缺省)和PTHREAD_CANCEL_DISABLE,
    分别表示收到信号后设为CANCLED状态和忽略CANCEL信号继续运行;old_state如果不为NULL则存入原来的Cancel状态以便恢复。  

    int pthread_setcanceltype(int type, int *oldtype)   
    设置本线程取消动作的执行时机,type由两种取值:PTHREAD_CANCEL_DEFFERED和PTHREAD_CANCEL_ASYCHRONOUS,仅当Cancel状态为Enable时有效,分别表示收到信号后继续运行至下一个取消点再退出和立即执行取消动作(退出);oldtype如果不为NULL则存入运来的取消动作类型值。  

    void pthread_testcancel(void)
    是说pthread_testcancel在不包含取消点,但是又需要取消点的地方创建一个取消点,以便在一个没有包含取消点的执行代码线程中响应取消请求.
    线程取消功能处于启用状态且取消状态设置为延迟状态时,pthread_testcancel()函数有效。
    如果在取消功能处处于禁用状态下调用pthread_testcancel(),则该函数不起作用。
    请务必仅在线程取消线程操作安全的序列中插入pthread_testcancel()。除通过pthread_testcancel()调用以编程方式建立的取消点意外,pthread标准还指定了几个取消点。测试退出点,就是测试cancel信号.


    取消点:
    线程取消的方法是向目标线程发Cancel信号,但如何处理Cancel信号则由目标线程自己决定,或者忽略、或者立即终止、或者继续运行至Cancelation-point(取消点),由不同的Cancelation状态决定。

    线程接收到CANCEL信号的缺省处理(即pthread_create()创建线程的缺省状态)是继续运行至取消点,也就是说设置一个CANCELED状态,线程继续运行,只有运行至Cancelation-point的时候才会退出。

    pthreads标准指定了几个取消点,其中包括:
    (1)通过pthread_testcancel调用以编程方式建立线程取消点。 
    (2)线程等待pthread_cond_wait或pthread_cond_timewait()中的特定条件。 
    (3)被sigwait(2)阻塞的函数 
    (4)一些标准的库调用。通常,这些调用包括线程可基于阻塞的函数。 
      
    缺省情况下,将启用取消功能。有时,您可能希望应用程序禁用取消功能。如果禁用取消功能,则会导致延迟所有的取消请求,
    直到再次启用取消请求。  
    根据POSIX标准,pthread_join()、pthread_testcancel()、pthread_cond_wait()、pthread_cond_timedwait()、sem_wait()、sigwait()等函数以及
    read()、write()等会引起阻塞的系统调用都是Cancelation-point,而其他pthread函数都不会引起Cancelation动作。
    但是pthread_cancel的手册页声称,由于LinuxThread库与C库结合得不好,因而目前C库函数都不是Cancelation-point;但CANCEL信号会使线程从阻塞的系统调用中退出,并置EINTR错误码,因此可以在需要作为Cancelation-point的系统调用前后调用pthread_testcancel(),从而达到POSIX标准所要求的目标.
    即如下代码段:
    pthread_testcancel();
    retcode = read(fd, buffer, length);
    pthread_testcancel();

    注意:
    程序设计方面的考虑,如果线程处于无限循环中,且循环体内没有执行至取消点的必然路径,则线程无法由外部其他线程的取消请求而终止。因此在这样的循环体的必经路径上应该加入pthread_testcancel()调用.


    取消类型(Cancellation Type)

    我们会发现,通常的说法:某某函数是 Cancellation Points,这种方法是容易令人混淆的。
    因为函数的执行是一个时间过程,而不是一个时间点。其实真正的 Cancellation Points 只是在这些函数中 Cancellation Type 被修改为 PHREAD_CANCEL_ASYNCHRONOUS 和修改回 PTHREAD_CANCEL_DEFERRED 中间的一段时间。

    POSIX的取消类型有两种,一种是延迟取消(PTHREAD_CANCEL_DEFERRED),这是系统默认的取消类型,即在线程到达取消点之前,不会出现真正的取消;另外一种是异步取消(PHREAD_CANCEL_ASYNCHRONOUS),使用异步取消时,线程可以在任意时间取消。


    线程终止的清理工作

    Posix的线程终止有两种情况:正常终止和非正常终止。
    线程主动调用pthread_exit()或者从线程函数中return都将使线程正常退出,这是可预见的退出方式;
    非正常终止是线程在其他线程的干预下,或者由于自身运行出错(比如访问非法地址)而退出,这种退出方式是不可预见的。

    不论是可预见的线程终止还是异常终止,都会存在资源释放的问题,在不考虑因运行出错而退出的前提下,如何保证线程终止时能顺利的释放掉自己所占用的资源,特别是锁资源,就是一个必须考虑解决的问题。
    最经常出现的情形是资源独占锁的使用:线程为了访问临界资源而为其加上锁,但在访问过程中被外界取消,如果线程处于响应取消状态,且采用异步方式响应,或者在打开独占锁以前的运行路径上存在取消点,则该临界资源将永远处于锁定状态得不到释放。外界取消操作是不可预见的,因此的确需要一个机制来简化用于资源释放的编程。

    在POSIX线程API中提供了一个pthread_cleanup_push()/ pthread_cleanup_pop()函数,
    对用于自动释放资源—从pthread_cleanup_push()的调用点到pthread_cleanup_pop()之间的程序段中的终止动作(包括调用pthread_exit()和取消点终止)都将执行pthread_cleanup_push()所指定的清理函数。

    API定义如下:
    void pthread_cleanup_push(void (*routine) (void *), void *arg)
    void pthread_cleanup_pop(int execute)

    pthread_cleanup_push()/pthread_cleanup_pop()采用先入后出的栈结构管理,void routine(void *arg)函数
    在调用pthread_cleanup_push()时压入清理函数栈,多次对pthread_cleanup_push() 的调用将在清理函数栈中形成一个函数链;
    从pthread_cleanup_push的调用点到pthread_cleanup_pop之间的程序段中的终止动作(包括调用pthread_exit()和异常终止,不包括return)
    都将执行pthread_cleanup_push()所指定的清理函数。

    在执行该函数链时按照压栈的相反顺序弹出。execute参数表示执行到 pthread_cleanup_pop()时
    是否在弹出清理函数的同时执行该函数,为0表示不执行,非0为执行;这个参数并不影响异常终止时清理函数的执行。

    pthread_cleanup_push()/pthread_cleanup_pop()是以宏方式实现的,这是pthread.h中的宏定义:

    复制代码
    #define pthread_cleanup_push(routine,arg)  
    { 
    struct _pthread_cleanup_buffer _buffer;  
    _pthread_cleanup_push (&_buffer, (routine), (arg));
    
    #define pthread_cleanup_pop(execute)  
    _pthread_cleanup_pop (&_buffer, (execute)); 
    }
    复制代码

    可见,pthread_cleanup_push()带有一个"{",而pthread_cleanup_pop()带有一个"}",因此这两个函数必须成对出现,且必须位于程序的同一级别的代码段中才能通过编译。

    在下面的例子里,当线程在"do some work"中终止时,将主动调用pthread_mutex_unlock(mut),以完成解锁动作。

    复制代码
    pthread_cleanup_push(pthread_mutex_unlock, (void*) &mut);
    pthread_mutex_lock(&mut);
    /* do some work */
    pthread_mutex_unlock(&mut);
    pthread_cleanup_pop(0);
    或者
    void cleanup(void *arg)
    {    
        pthread_mutex_unlock(&mutex);
    }
    
    void* thread0(void* arg)
    {    
        pthread_cleanup_push(cleanup, NULL); // thread cleanup handler    p
        thread_mutex_lock(&mutex);    
        pthread_cond_wait(&cond, &mutex);    
        pthread_mutex_unlock(&mutex);    
        pthread_cleanup_pop(0);    
        pthread_exit(NULL);
    }
    复制代码

    参考博文:
    取消线程及清理工作
    http://blog.sina.com.cn/s/blog_66fb0c830100y9hb.html

    (线程终止的清理工作)
    http://www.cnblogs.com/mydomain/archive/2011/08/15/2139826.html 

    (一个 pthread_cancel 引起的线程死锁)
    http://www.cnblogs.com/mydomain/archive/2011/08/15/2139830.html

    http://www.cnblogs.com/mydomain/archive/2011/08/15/2139850.html

    自己sina博文(JAVA多线程相关介绍)
    http://blog.sina.com.cn/s/blog_8da6362401013rcl.html

  • 相关阅读:
    pat03-树1. 二分法求多项式单根(20)
    pat05-图1. List Components (25)
    pat06-图4. Saving James Bond
    pat05-图3. 六度空间 (30)
    pat05-图2. Saving James Bond
    pat04-树9. Path in a Heap (25)
    pat04-树8. Complete Binary Search Tree (30)
    pat04-树7. Search in a Binary Search Tree (25)
    pat04-树5. File Transfer (25)
    Two Sum
  • 原文地址:https://www.cnblogs.com/24zyt/p/6881366.html
Copyright © 2020-2023  润新知