pthread_kill

          Linux杀死子线程可以用pthread_cancel。但是Google认为pthread_cancel存在资源泄漏风险,所以在Android中禁止使用pthread_cancel,
应该使用pthread_kill。
          pthread_kill可不是kill,而是向线程发送signal。还记得signal吗,大部分signal的默认动作是终止进程的运行,所以,我们才要用signal()去抓信号并加上处理函数。如果线程代码内不做处理,则按照信号默认的行为影响整个进程,也就是说,如果你给一个线程发送了SIGQUIT,但线程却没有实现signal处理函数,则整个进程退出。pthread_kill(threadid, SIGKILL)也一样,杀死整个进程。如果要获得正确的行为,就需要在线程内实现signal(SIGKILL,sig_handler)了。所以,如果int sig的参数不是0,那一定要清楚到底要干什么,而且一定要实现线程的信号处理函数,否则,就会影响整个进程。
         如果int sig是0呢,这是一个保留信号,一个作用是用来判断线程是不是还活着。

#include <stdio.h>
#include <unistd.h>
#include <signal.h>
#include <errno.h>
#include <pthread.h>
#include <stdlib.h>     // sleep() 函数

static void signal_handler_fun(int sig_num)
{
    if (sig_num == SIGUSR1) {
        printf("thread quit by SIGQUIT\n");
        (void)pthread_exit((void *)1);
    }

    return;
}

//线程执行的函数
void * thread_Fun(void * arg) {
    if (signal(SIGUSR1, signal_handler_fun) == SIG_ERR) {
        printf("Fail to register signal\n");
    }

    printf("新建线程开始执行\n");

    while(1);
}

int main()
{
    pthread_t myThread;
    void * mess;
    int value;
    int res;
    //创建 myThread 线程
    res = pthread_create(&myThread, NULL, thread_Fun, NULL);
    if (res != 0) {
        printf("线程创建失败\n");
        return 0;
    }
    sleep(2);
    //向 myThread 线程发送 Cancel 信号
    //res = pthread_cancel(myThread);
    int kill_rc = pthread_kill(myThread, 0);
    if (kill_rc == 0) {
        printf("geekbench thread exists and kill it.\n");
        pthread_kill(myThread, SIGUSR1);
    }

    printf("sleep.\n");
    sleep(3);
    printf("main exit.\n");
    return 0;
}

79@u10-121-79-98:~/test$ gcc thread.c -o main -lpthread
79@u10-121-79-98:~/test$ ./main
新建线程开始执行
geekbench thread exists and kill it.
sleep.
thread quit by SIGQUIT
main exit.

posted @ 2023-07-08 09:22  牧 天  阅读(768)  评论(0)    收藏  举报