实现多线程下msgsnd,msgrcv的超时

8 篇文章 0 订阅

msgsnd和msgrcv,只有在一下三种情况下,才会退出阻塞状态

  1. 写或读消息队列成功
  2. 对应的消息队列被删除
  3. 信号中断

为了做到超时退出阻塞状态,一般常用的做法是通过alarm,触发信号中断,但是只能是在一个线程的情况下执行。
多线程下调用alarm,SIGALRM信号会传递给其中一个线程(我测试下是传递给主线程…)

我的做法是,写个定时器,由单独的一个线程执行定时器,执行msgsnd和msgrcv的线程需要注册相应的事件到定时器中,由定时器线程通过pthread_kill,向对应线程发送信号,使得msgsnd和msgrcv退出阻塞状态。

2021/09/15 更新

旧方法

  1. 定时器线程用rbtree + select作为定时器(ms);
  2. msgsnd/msgrcv前,注册定时器任务,用pipe通知定时器线程;
  3. 定时器线程超时后,用pthread_kill中断msgsnd/msgrcv线程的阻塞状态。

新方法

  1. 定时器线程用rbtree + pthread_cond_wait作为定时器(ns);
  2. msgsnd/msgrcv前,注册定时器任务,用pthread_cond_signal通知定时器线程;
  3. 定时器线程超时后,用pthread_kill中断msgsnd/msgrcv线程的阻塞状态。

代码在GitHub上
https://github.com/LiuYuguang/MessageQueue

  • 2
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
`msgsnd`和`msgrcv`是Linux消息队列的两个主要函数,它们可以用于进程间通信。下面是它们的用法: 1. `msgsnd`函数 `msgsnd`函数用于将消息发送到消息队列中,它的语法如下: ```c int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg); ``` 参数说明: - `msqid`:消息队列的ID。 - `msgp`:指向要发送的消息的指针。 - `msgsz`:消息的大小(字节数)。 - `msgflg`:控制发送操作的标志。 返回值: - 成功:返回0。 - 失败:返回-1,errno变量被设置为相应的错误代码。 2. `msgrcv`函数 `msgrcv`函数用于从消息队列中接收消息,它的语法如下: ```c ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg); ``` 参数说明: - `msqid`:消息队列的ID。 - `msgp`:指向要接收消息的指针。 - `msgsz`:接收消息缓冲区的大小(字节数)。 - `msgtyp`:指定要接收的消息类型。如果为0,则接收队列中的第一条消息。 - `msgflg`:控制接收操作的标志。 返回值: - 成功:返回接收到的消息的字节数。 - 失败:返回-1,errno变量被设置为相应的错误代码。 注意事项: - 在使用`msgsnd`和`msgrcv`函数时,要保证消息队列已经被创建和初始化。 - 消息的类型是一个长整型数值,可以自定义。接收消息时,如果指定了消息类型,则只接收该类型的消息。 - 在发送和接收消息时,要注意消息的大小不能超过消息队列的最大容量。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值