线程的退出


2.线程的退出
2.1 线程的正常退出处理:

    1.可预知的错误发生时,应对处理:使整个程序stop的方法
    设置全局的开关 ,在主进程和线程中使用【线程中使用extern 进行引用】,当主进程或线程中的开关发生变化,要求退出时,所有线程退出,主进程调用函数pthread_join()阻塞式的等待所有线程的退出。

    2.可预知的错误发生时,应对处理:使整个程序stop的方法
    对于接受线程:可以内部发送控制信息【退出线程消息】 ,使各个接受线程退出【建议使用1.方法】

2.2 线程的要求退出处理:
   1.主进程使用pthread_cancel(pthread_t thread)函数使线程退出。相当于发送信号使线程退出。
   但一旦使用pthread_cancel函数就必须的考虑互斥锁的问题,因为有可能发生死锁问题
当调用pthread_mutex_lock函数后,在调用pthread_mutex_unlock之前使用了pthread_cancel函数,就会造成死锁。
【建议】在编写代码时,不要使用pthread_cancel函数。

*****************************************************
    一般来说,对一段运行代码进行加锁然后解锁,如下所示:
  pthread_mutex_lock(&mutex);

  //运行代码;

  pthread_mutex_unlock(&mutex);

  如果在运行代码这块发生错误,有异常,导致这个线程异常退出,那么怎么办,pthread_unlock没有得到调用,那么这个锁资源没有解锁。可以用下面的方法修改。

  pthread_cleanup_push(pthread_mutex_unlock, (void *) &mutex);

  pthread_mutex_lock(&mutex);

  /* do some work */

  pthread_mutex_unlock(&mutex);

  pthread_cleanup_pop(0);

  这样假如运行代码发生错误时没有调用到解锁,pthread_cleanup_up会自动来调用,参数为0表示不执行push进来的函数。

  但是如果是异常错误的话,这个参数并不影响异常终止时清理函数的执行。

  必须要注意的是,如果线程处于PTHREAD_CANCEL_ASYNCHRONOUS状态,上述代码段就有可能出错,因为CANCEL事件有可能在pthread_cleanup_push()和pthread_mutex_lock()之间发生,或者在pthread_mutex_unlock()和pthread_cleanup_pop()之间发生,从而导致清理函数unlock一个并没有加锁的mutex变量,造成错误。因此,在使用清理函数的时候,都应该暂时设置成PTHREAD_CANCEL_DEFERRED模式。为此,POSIX的Linux实现中还提供了一对不保证可移植的pthread_cleanup_push_defer_np()/pthread_cleanup_pop_defer_np()扩展函数,功能与以下代码段相当:

  { int oldtype;

  pthread_setcanceltype(PTHREAD_CANCEL_DEFERRED, &oldtype);

  pthread_cleanup_push(routine, arg);

  ...

  pthread_cleanup_pop(execute);

  pthread_setcanceltype(oldtype, NULL);

  }

  +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++

  设置退出类型pthread_setcanceltype

  #include <pthread.h>

  int pthread_setcanceltype(int type, int *oldtype);

  返回值:函数成功返回0。任何其他返回值都表示错误。

  将线程退出类型设置为延迟类型或异步类型。参数type的取值为PTHREAD_CANCEL_DEFERRED或PTHREAD_CANCEL_ASYNCHRONOUS。

  当一个线程被创建后,缺省值是延迟类型。在异步方式下,线程可以在执行的任何时候被退出。

你可能感兴趣的:(JOIN,thread,linux,null,扩展,asynchronous)