项目中遇到的pthread_cond_signal/pthread_cond_timedwait使用问题

问题现象

项目上遇到一个问题,报文交互经常延时过高,客户非常不满。

初步抓包分析

通过抓包发现问题出在从物理口收包到我们模块处理时。也就是说不是网络中异常导致的延时过高。是我们处理过程中产生的。但是但从抓包很难确认延时产生在什么位置。需要进一步确认。

报文打点确认

于是对报文进行了处理,使其记录timestamp,然后在各个关键位置判断当前时间和报文timestamp的差值,如果超过一定范围,则通过log打印出来。
通过打点,确认问题发生在 业务模块将 报文放入队列,到主线程从队列取包处理时。

这部分的简单逻辑如下:

  • 模块从物理口收到报文后,会将其放在一个队列中,然后通过 pthread_cond_signal 通知主线程进行处理。
  • 主线程loop通过pthread_cond_timedwait等待信号或者超时。如果是信号唤醒,则从队列中取包进行处理。如果是超时,则处理一些异步事件。处理完毕后继续通过 pthread_cond_timedwait 等待下一次。

曾怀疑过是不是 pthread_cond_timedwait 唤醒太慢导致,增加debug log发现不是。那为什么报文到达队列后,等待很久才被主线程取出处理呢?

恍然大悟

回头又审视 pthread_cond_signal 函数,manual上写:

The pthread_cond_signal() function shall unblock at least one of the threads that are blocked on the specified condition variable cond (if any  threads  are  blocked  on cond).

pthread_cond_signal唤醒在制定cond条件上被阻塞的线程。如果没有线程被block ,那么pthread_cond_signal就相当于空操作了。
于是明白了为什么入队后的报文没有及时被处理:

  • 报文入队后,pthread_cond_signal去唤醒主线程。与此同时主线程的pthread_cond_timedwait 已经超时唤醒。pthread_cond_signal信号无效忽略。
  • 因为主线程pthread_cond_timedwait是超时唤醒,并未进行队列取包处理,只是处理异步事件。处理完毕后调用pthread_cond_timedwait等待下一次。
  • 直到某个报文入队,pthread_cond_signal去唤醒主线程,且主线程正在pthread_cond_timedwait休眠等待尚未唤醒和超时。此时主线程被signal信号唤醒,从队列中取包处理。此时之前队列中积压的报文都得到了处理,但是较早入队的报文处理时间间隔就会很高。

修复操作

知道原因后,修复起来就简单了。之前主线程对信号唤醒还是超时唤醒进行了区分操作(超时唤醒假定了队列中没有包问,不处理报文队列)。实际上不需要,无论哪种唤醒都对队列进行取包处理。这样就能避免该问题的发生。

后记

之前只关注了 pthread_cond_signal 的惊群问题,没注意到pthread_cond_timedwait唤醒间隙,pthread_cond_signal无效问题 。

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
pthread_cond_timedwait函数是POSIX线程库用于等待条件变量的函数之一。它可以在指定的时间内等待条件变量的状态发生变化。 下面是一个使用pthread_cond_timedwait函数的例子: ```c #include <stdio.h> #include <pthread.h> #include <unistd.h> pthread_mutex_t mutex; pthread_cond_t cond; void* thread_func(void* arg) { pthread_mutex_lock(&mutex); printf("Thread waiting...\n"); struct timespec timeout; clock_gettime(CLOCK_REALTIME, &timeout); timeout.tv_sec += 5; // 设置等待时间为5秒 int result = pthread_cond_timedwait(&cond, &mutex, &timeout); if (result == 0) { printf("Thread woken up!\n"); } else if (result == ETIMEDOUT) { printf("Thread timed out!\n"); } else { printf("Thread wait error!\n"); } pthread_mutex_unlock(&mutex); return NULL; } int main() { pthread_t thread; pthread_mutex_init(&mutex, NULL); pthread_cond_init(&cond, NULL); pthread_create(&thread, NULL, thread_func, NULL); sleep(2); // 等待2秒,确保子线程已经开始等待 pthread_mutex_lock(&mutex); printf("Main thread signaling...\n"); pthread_cond_signal(&cond); pthread_mutex_unlock(&mutex); pthread_join(thread, NULL); pthread_mutex_destroy(&mutex); pthread_cond_destroy(&cond); return 0; } ``` 在这个例子,我们创建了一个子线程,子线程会在条件变量上等待。主线程等待2秒后,通过调用pthread_cond_signal函数来通知子线程条件变量的状态发生了变化。子线程在等待时间到达或者收到信号时会被唤醒,并打印相应的信息。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

浮沉飘摇

码字不易,打赏随意。

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值