Chinaunix首页 | 论坛 | 博客
  • 博客访问: 7888721
  • 博文数量: 701
  • 博客积分: 2150
  • 博客等级: 上尉
  • 技术积分: 13233
  • 用 户 组: 普通用户
  • 注册时间: 2011-06-29 16:28
个人简介

天行健,君子以自强不息!

文章分类

全部博文(701)

文章存档

2019年(2)

2018年(12)

2017年(76)

2016年(120)

2015年(178)

2014年(129)

2013年(123)

2012年(61)

分类: PERL

2013-03-12 14:57:40

一、ALARM方式
示例程序:
#!/usr/bin/perl

$timeout = 2 ;
while(1)
{
  eval {
    local $SIG{ALRM} = sub{print "sorry,time out.please try again\n"} ;
    alarm $timeout ; #启动定时器


    # 实际操作
    print "hello world!\n" ; 
    sleep 3 ;


    alarm 0 ;        #取消定时器
  } ;
  die $@ if $@ ;
}


说明:
将实际操作放在eval块中。

首先,初始化一个本地化的ALRM信号处理器,它位于特殊的%SIG哈希里。
假如触发了该处理器,它会调用die()并且中断eval块。然后可以根据$SIG{ALRM}信号处理程序来做相应的处理。
大多数情况下,你可能需要向用户报告,操作已超时。

实际的操作放在2个alarm()调用之间。
第一个alarm开始计时器,第二个取消它。这里计时器运行10秒。
假如在指定的时间里,第二个alarm没有发生,则产生SIGALRM信号,存储在$SIG{ALRM}里的处理器被调用。
假如在第二个alarm在设定的秒内完成,alarm时钟会终止,eval块成功返回,不会触发ALRM处理器。

注意:
在给定时间内,仅仅一个计时器可用。alarm()的返回值是剩余时间的数量。
所以实际上可利用这点来粗略的估计执行时间。

二、父子进程方式
通过fork(), 产生子进程,让子进程执行实际操作;
父进程中设置时间门限sleep();
若子进程成功执行则会自动激发$SIG{CHLD}消息,在消息处理里收集状态;
若不成功,则在父进程中使用kill杀死子进程,
此时也会激发$SIG{CHLD}消息,调用waitpid收集僵尸。并处理状态。 

示例:
用Perl 给命令加上超时退出的功能

#! /usr/bin/env perl
use POSIX qw(strftime WNOHANG);

#check input
my $timeout = shift @ARGV;
my ($secs) = $timeout =~ /--timeout=(\d+)$/;
unless($secs)
{
  print "Usage: ./timeout --timeout=[SECONDS] [COMMAND] \n";
  exit -1;
}

#fork and exec
my $status = 0;
$SIG{CHLD} = sub 

  while(waitpid(-1,WNOHANG)>0)
  { 
    $status = -1 unless $? == 0; 
    exit $status;
  } 
};

$0 = 'timeout hacked ' . $ARGV[0];
defined (my $child = fork);
if($child == 0)
{
  my $cmd = join ' ', @ARGV;
  exec($cmd);
}
$SIG{TERM} = sub { kill TERM => $child };
$SIG{INT} = sub { kill INT => $child };

#kill when timeout
sleep $secs;
$status = -1;
kill TERM => $child;
sleep 1 and kill INT => $child if kill 0 => $child;
sleep 1 and kill KILL => $child if kill 0 => $child;
exit $status;

如下调用, 就可以让任意的命令超时退出了(这里执行的命令是“sleep 500”)
./timeout.pl --timeout=3 sleep 500

参数文章:
1. http://blog.yikuyiku.com/?tag=perl
2. http://fuzhong1983.blog.163.com/blog/static/16847052010101624315872/


阅读(1963) | 评论(0) | 转发(0) |
给主人留下些什么吧!~~