当前位置: 技术问答>linux和unix
linux多线程里的定时问题
来源: 互联网 发布时间:2016-02-20
本文导语: 我需要在一个程序里开个定时期,如在5秒后做一件事情,我用了alarm(5),然而我的程序母体是个多线程程序,需要在多处实现此机制,而alarm函数有两个问题: 1 alarm函数在一个进程里一个时刻只能有一个存在,需要更...
我需要在一个程序里开个定时期,如在5秒后做一件事情,我用了alarm(5),然而我的程序母体是个多线程程序,需要在多处实现此机制,而alarm函数有两个问题:
1 alarm函数在一个进程里一个时刻只能有一个存在,需要更好的solution并想知道问题2的原因
2 在同一进程里我试图在前后多次使用,第一次可以成功,以后就不能成功了,不知为何?
我是这样做的:
alarm(5);
signal(SIGALRM,handler);
...//这里是一个可能阻塞很久的函数,如gethostbyname或其他I/O函数
...
alarm(0);
...
alarm(5);
signal(SIGALRM,handler);
...//另外一处阻塞函数
...
alarm(0);
希望有相关经历、经验或研究的高手不吝赐教,小弟感激涕零...
1 alarm函数在一个进程里一个时刻只能有一个存在,需要更好的solution并想知道问题2的原因
2 在同一进程里我试图在前后多次使用,第一次可以成功,以后就不能成功了,不知为何?
我是这样做的:
alarm(5);
signal(SIGALRM,handler);
...//这里是一个可能阻塞很久的函数,如gethostbyname或其他I/O函数
...
alarm(0);
...
alarm(5);
signal(SIGALRM,handler);
...//另外一处阻塞函数
...
alarm(0);
希望有相关经历、经验或研究的高手不吝赐教,小弟感激涕零...
|
当系统发生信号时,你的 gethostbyname()函数能立即返回吗???如果任何信号都可以导致 gethostbyname()函数返回的话,你这个策略是行不通的,除非绕过这种机制,如果信号无法让gethostbyname()函数立即返回,你又想随时让这个函数返回的话,这也是行不通的......你想想可不可以采用其它方案来绕过这种问题??????
比如你要在三个线程中同时使用gethostbyname()函数,假定任何信号都能让gethostbyname()函数立即返回,那么你无论如何也做不到让每个信号去唤醒单个线程中的特定的gethostbyname()函数,而不去唤醒另一个线程中的gethostbyname()函数.假定任何信号都不能让gethostbyname()函数立即返回,那么你想要实现的功能就无法实现了.所以这无论如何都是行不通的,还是想想别的办法,比如可否使用非阻塞版本的gethostbyname()函数(或者能实现相同功能,但又不会阻塞的另一个函数)......
比如你要在三个线程中同时使用gethostbyname()函数,假定任何信号都能让gethostbyname()函数立即返回,那么你无论如何也做不到让每个信号去唤醒单个线程中的特定的gethostbyname()函数,而不去唤醒另一个线程中的gethostbyname()函数.假定任何信号都不能让gethostbyname()函数立即返回,那么你想要实现的功能就无法实现了.所以这无论如何都是行不通的,还是想想别的办法,比如可否使用非阻塞版本的gethostbyname()函数(或者能实现相同功能,但又不会阻塞的另一个函数)......
|
如果你想通过siglongjmp()/longjmp() 这种方式来达到退出gethostbyname()函数的目的,而不是通过信号中断去gethostbyname()函数并让其返回的话,可以采用另一种方案:
1.每个线程都保存一个需要恢复的线程上下文
2.每个线程向你自己的定时器管理链表注册回调函数(此函数就是恢复到第一步中所保存的上下文中)
3.由于任何信号都是与进程关联,而不是跟线程关联,所以信号可能是发生在任何线程当中的,你需要一定的策略,以保证只有当自己这个线程得到调度的时候,才去恢复自己的上下文,而gethostbyname()函数却阻塞在这个线程当中,也就是说我们这里需要的条件根本无法得到满足
4.由于需要的条件根本无法得到满足,那得要换一种策略,比如用两个线程来实现一个gethostbyname()的功能,一个线程来实现自己线程中的逻辑处理,另一个线程只用来实现gethostbyname()功能.gethostbyname()函数所在的线程只做一件事情:那就是不停地去gethostbyname,从来不去打断它,它愿意阻塞到什么时候就阻塞到什么时候,我只需要另一个线程来实成时间控制,比如在规定的时间内gethostbyname的功能还未实现,我就做自己相应的逻辑处理,也就是说换一种方案来解决LZ的问题,而不再局限于用信号或者定时器来处理这个问题.这就相当于用两个线程来实现非阻塞版本的gethostbyname()功能
1.每个线程都保存一个需要恢复的线程上下文
2.每个线程向你自己的定时器管理链表注册回调函数(此函数就是恢复到第一步中所保存的上下文中)
3.由于任何信号都是与进程关联,而不是跟线程关联,所以信号可能是发生在任何线程当中的,你需要一定的策略,以保证只有当自己这个线程得到调度的时候,才去恢复自己的上下文,而gethostbyname()函数却阻塞在这个线程当中,也就是说我们这里需要的条件根本无法得到满足
4.由于需要的条件根本无法得到满足,那得要换一种策略,比如用两个线程来实现一个gethostbyname()的功能,一个线程来实现自己线程中的逻辑处理,另一个线程只用来实现gethostbyname()功能.gethostbyname()函数所在的线程只做一件事情:那就是不停地去gethostbyname,从来不去打断它,它愿意阻塞到什么时候就阻塞到什么时候,我只需要另一个线程来实成时间控制,比如在规定的时间内gethostbyname的功能还未实现,我就做自己相应的逻辑处理,也就是说换一种方案来解决LZ的问题,而不再局限于用信号或者定时器来处理这个问题.这就相当于用两个线程来实现非阻塞版本的gethostbyname()功能