免费注册 查看新帖 |

Chinaunix

  平台 论坛 博客 文库
最近访问板块 发新帖
查看: 2355 | 回复: 4
打印 上一主题 下一主题

问一个信号调用函数和线程互斥的问题? [复制链接]

论坛徽章:
0
跳转到指定楼层
1 [收藏(0)] [报告]
发表于 2012-05-12 23:00 |只看该作者 |倒序浏览
本帖最后由 wangjun0629 于 2012-05-12 23:02 编辑

初始化
pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;

我的程序是单线程,我想在线程和信号处理函数之间做一个同步。

a函数{
pthread_mutex_lock(&mutex)
sleep(10)
......
pthread_mutex_unlock(&mutex)
}


信号处理函数
signel(int sig)
{
pthread_mutex_lock(&mutex)
.......
pthread_mutex_unlock(&mutex)
}

实际发现,当我发送信号给线程时,如果线程当时在调用到sleep的话。
程序就卡死了。 这是为什么呢?pthread_mutex_lock里面不能调用sleep吗?
如果这个方法不行,有其他方法来实现同步吗?


论坛徽章:
0
2 [报告]
发表于 2012-05-13 11:30 |只看该作者
没人能回答么。

论坛徽章:
0
3 [报告]
发表于 2012-05-14 17:48 |只看该作者
  1 #include <stdio.h>
  2 #include <signal.h>
  3 #include <pthread.h>
  4
  5 pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
  6
  7 void sig_all(int signum)
  8 {
  9     pthread_mutex_lock(&mutex);
10     printf("sig_all\n");
11     pthread_mutex_unlock(&mutex);
12 }
13
14 int main()
15 {
16     if (SIG_ERR == signal(SIGINT, sig_all))
17         perror("signal");
18     pthread_mutex_lock(&mutex);
19     printf("__main__\n");
20     sleep(10);
21     printf("__main__\n");
22     pthread_mutex_unlock(&mutex);
23
24     return 0;
25 }
这样写没啥问题呀。单线程不会出问题的呀。
  1 #include <stdio.h>
  2 #include <signal.h>
  3 #include <pthread.h>
  4
  5 pthread_mutex_t mutex = PTHREAD_MUTEX_INITIALIZER;
  6
  7 void sig_all(int signum)
  8 {
  9     printf("signal...\n");
10     pthread_mutex_lock(&mutex);
11     printf("sig_all\n");
12     pthread_mutex_unlock(&mutex);
13 }
14
15 void* worker(void* args)
16 {
17     pthread_mutex_lock(&mutex);
18     printf("__worker__\n");
19     sleep(10);
20     printf("__worker__\n");
21     pthread_mutex_unlock(&mutex);
22     return (void*)0;
23 }
24
25 int main()
26 {
27     if (SIG_ERR == signal(SIGINT, sig_all))
28         perror("signal");
29     pthread_t tid;
30     if (0 != pthread_create(&tid, NULL, worker, NULL))
31         perror("pthread_create");
32
33     pthread_join(tid, NULL);
34     return 0;
35 }
上面的会出问题。问题在死锁了。进程中的子线程和主线程争夺锁死锁了。
我真不知道你上面测试的单线程怎么会出问题,我觉得只有一种可能就是你的系统里pthread_mutex_t默认不是可重入的。
据我对信号的了解,当进行系统调用的时候检测到信号,系统会返回到信号处理函数,信号处理函数结束之后再返回到进程中系统调用的位置。所以sleep的时候发送信号,返回到信号处理函数,信号处理函数获取锁,但是锁已经被另一个线程占用了。所以需要等待另一个线程,但是另一个线程需要等待信号处理函数执行完成之后才能返回。因此发生死锁。

论坛徽章:
0
4 [报告]
发表于 2013-07-30 10:35 |只看该作者
虽然是单线程,但是在涉及到1,信号处理; 2,非递归mutex的时候,是很容易出现死锁的。
原因在于,当线程在sleep的时候获取到了锁,但是当信号发生的时候,内核进行信号递送,唤醒了sleep 操作,该线程来负责响应信号处理函数。

此时,由于锁在sleep之前已经获得了,再次在信号处理函数中获取非递归锁,即导致自死锁。

解决办法,可以修改锁的属性为递归锁。

论坛徽章:
17
处女座
日期:2013-08-27 09:59:352015亚冠之柏太阳神
日期:2015-07-30 10:16:402015亚冠之萨济拖拉机
日期:2015-07-29 18:58:182015年亚洲杯之巴勒斯坦
日期:2015-03-06 17:38:17摩羯座
日期:2014-12-11 21:31:34戌狗
日期:2014-07-20 20:57:32子鼠
日期:2014-05-15 16:25:21亥猪
日期:2014-02-11 17:32:05丑牛
日期:2014-01-20 15:45:51丑牛
日期:2013-10-22 11:12:56双子座
日期:2013-10-18 16:28:17白羊座
日期:2013-10-18 10:50:45
5 [报告]
发表于 2013-07-30 10:47 |只看该作者
回复 1# wangjun0629


    这么奇怪的需求,signal handler和主线程不可能同时执行的,也就是说不存在资源竞争问题。为什么还要同步呢?
您需要登录后才可以回帖 登录 | 注册

本版积分规则 发表回复

  

北京盛拓优讯信息技术有限公司. 版权所有 京ICP备16024965号-6 北京市公安局海淀分局网监中心备案编号:11010802020122 niuxiaotong@pcpop.com 17352615567
未成年举报专区
中国互联网协会会员  联系我们:huangweiwei@itpub.net
感谢所有关心和支持过ChinaUnix的朋友们 转载本站内容请注明原作者名及出处

清除 Cookies - ChinaUnix - Archiver - WAP - TOP