pthread_cond_timedwait()用于取消冗长的任务

时间:2014-03-26 16:10:33

标签: c++ linux multithreading

我有一种情况,如果需要太多的话,我想取消一个帖子。为此,我使用第二个线程等待第一个线程完成,但不超过几秒钟。 pthread_cond_timedwait()函数似乎完全适合我的使用场景,但它似乎没有像我预期的那样表现。更具体地说,即使pthread_cond_timedwait()函数返回ETIMEDOUT,它也会在它应该取消的线程完成之后才会这样做,这会破坏整个目的。

这是我的测试代码:

    #include <unistd.h>
    #include <stdlib.h>
    #include <errno.h>
    #include <iostream>
    #include <cstring>

    #define WAIT_INTERVAL 5
    #define THREAD_SLEEP 10

    pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
    pthread_cond_t condition = PTHREAD_COND_INITIALIZER;

    pthread_t t1;
    pthread_t t2;

    void* f1(void*);
    void* f2(void*);

    int main()
    {
        pthread_create(&t1, NULL, &f1, NULL);
        pthread_create(&t2, NULL, &f2, NULL);

        pthread_join(t1, NULL);
        pthread_join(t2, NULL);

        std::cout << "Thread(s) successfully finished" << std::endl << std::flush;

        exit(EXIT_SUCCESS);
    }

    void* f1(void*)
    {
        pthread_mutex_lock(&mutex);
        timespec ts = {0};
        clock_gettime(CLOCK_REALTIME, &ts);
        ts.tv_sec += WAIT_INTERVAL;
        std::cout << __FUNCTION__ << ": Waiting for at most " << WAIT_INTERVAL << " seconds starting now" << std::endl << std::flush;
        int waitResult = pthread_cond_timedwait(&condition, &mutex, &ts);
        if (waitResult == ETIMEDOUT)
        {
            std::cout << __FUNCTION__ << ": Timed out" << std::endl << std::flush;
            int cancelResult = pthread_cancel(t2);
            if (cancelResult)
            {
                std::cout << __FUNCTION__ << ": Could not cancel T2 : " << strerror(cancelResult) << std::endl << std::flush;
            }
            else
            {
                std::cout << __FUNCTION__ << ": Cancelled T2" << std::endl << std::flush;
            }
        }
        std::cout << __FUNCTION__ << ": Finished waiting with code " << waitResult << std::endl << std::flush;
        pthread_mutex_unlock(&mutex);
    }

    void* f2(void*)
    {
        pthread_mutex_lock(&mutex);
        std::cout << __FUNCTION__ << ": Started simulating lengthy operation for " << THREAD_SLEEP << " seconds" << std::endl << std::flush;
        sleep(THREAD_SLEEP);
        std::cout << __FUNCTION__ << ": Finished simulation, signaling the condition variable" << std::endl << std::flush;
        pthread_cond_signal(&condition);
        pthread_mutex_unlock(&mutex);
    }

我从上面的代码得到的输出是:

    f1: Waiting for at most 5 seconds starting now
    f2: Started simulating lengthy operation for 10 seconds
    f2: Finished simulation, signaling the condition variable
    f1: Timed out
    f1: Could not cancel T2 : No such process
    f1: Finished waiting with code 110
    Thread(s) successfully finished

鉴于这是我第一次使用POSIX线程,我想我错过了一些可能非常明显的东西。

我已经阅读了很多关于此的教程,文章和答案,但没有一个涵盖我的用例,没有提供任何提示。

请注意,为简洁起见,我删除了一些处理pthread_cond_timedwait手册中提到的谓词的代码,因为这并没有改变行为中的任何内容。

我在CentOS 6.5机器上使用POSIX线程。我的开发和测试环境: 2.6.32-431.5.1.el6.centos.plus.x86_64#1 SMP x86_64 x86_64 x86_64 GNU / Linux g ++(GCC)4.4.7 20120313(Red Hat 4.4.7-4)

编译命令:g ++ -o executable_binary -pthread -lrt source_code.cpp

2 个答案:

答案 0 :(得分:4)

编辑:我首先建议不要使用pthread_cond_timedwait,但我认为在这种情况下它是可以的,所以第一个线程不会比需要的时间更长,虽然不是检查返回值,我会检查一个&#39 ;完成&#39;标志,由第二个线程设置并受互斥锁保护。

您的示例中的问题是第一个线程占用互斥锁,pthread_cond_timedwait()调用释放互斥锁。然后由第二个线程执行,从而阻塞第一个线程,直到第二个线程在结束时释放互斥锁。

答案 1 :(得分:1)

您正在使用

设置两个主题
 pthread_create(&t1, NULL, &f1, NULL);
 pthread_create(&t2, NULL, &f2, NULL);

我将使用线程t1取消t2而不是仅加入t1:在线程pthread_cancel(t2)中添加一条在计时器结束后读取t2的行。

这将向t1发送消息,告知其终止。您可以保留两个连接语句,这意味着t2将耐心等待{{1}}完成其死亡之痛,然后继续:)

如果您需要更多信息,请告诉我们:)