有没有办法构建一个时间计数器,使脚本的一部分能够运行,只要它打开?例如,我有以下代码:
for my $i (0 .. $QUOTA-1) {
build_dyna_file($i);
comp_simu_exe;
bin2txt2errormap($i);
}
理论上我想运行这个循环3分钟,即使循环指令尚未完成,它仍然应该在3分钟后突破循环。
实际上程序会打开一个时间计数器窗口,它与脚本的一部分并行工作(每次调用它)。
此外,子调用'comp_simu_exe'运行外部模拟器(在shell中),当超时结束时 - 此过程也必须被杀死(不要假设在一段时间后返回)。
sub comp_simu_exe{
system("simulator --shell");
}
系统函数调用死机问题之间是否存在任何关联?
答案 0 :(得分:12)
您可以设置一个警报,该警报将在指定的秒数后突破您的代码:
eval {
local $SIG{ ALRM } = sub { die "TIMEOUT" };
alarm 3 * 60;
for (my $i = 0 ; $i <$QUOTA ; $i++) {
build_dyna_file($i);
comp_simu_exe;
bin2txt2errormap($i);
}
alarm 0;
};
if ( $@ && $@ =~ m/TIMEOUT/ ) {
warn "operation timed out";
}
else {
# somebody else died
alarm 0;
die $@;
}
或者,如果你真的需要循环运行至少三次,无论这可能需要多长时间:
eval {
my $t0 = time;
local $SIG{ ALRM } = sub { die "TIMEOUT" };
for (my $i = 0 ; $i <$QUOTA ; $i++) {
build_dyna_file($i);
comp_simu_exe;
bin2txt2errormap($i);
if ( $i == 3 ) {
my $time_remaining = 3 * 60 - time - $t0;
alarm $time_remaining if $time_remaining > 0;
}
}
alarm 0;
};
答案 1 :(得分:5)
这是第二个答案,它涉及超时第二个过程的情况。使用这种情况启动外部程序并确保它不会花太长时间:
my $timeout = 180;
my $pid = fork;
if ( defined $pid ) {
if ( $pid ) {
# this is the parent process
local $SIG{ALRM} = sub { die "TIMEOUT" };
alarm 180;
# wait until child returns or timeout occurs
eval {
waitpid( $pid, 0 );
};
alarm 0;
if ( $@ && $@ =~ m/TIMEOUT/ ) {
# timeout, kill the child process
kill 9, $pid;
}
}
else {
# this is the child process
# this call will never return. Note the use of exec instead of system
exec "simulator --shell";
}
}
else {
die "Could not fork.";
}
答案 2 :(得分:3)
在Perl中处理这样的超时的方法是alarm
函数。在您传入数据的秒数后,它会将信号ALRM
发送到您的进程。如果您尝试设置呼叫sleep
超时的代码,请小心,因为它们在许多平台上不能很好地混合。基本结构如下所示:
#start a block eval to stop the die below from ending the program
eval {
#set the signal handler for the ALRM signal to die if it is run
#note, the local makes this signal handler local to this block
#eval only
local $SIG{ALRM} = sub { die "timeout\n" };
alarm $wait; #wait $wait seconds and then send the ALRM signal
#thing that could take a long time
alarm 0; #turn off the alarm (don't send the signal)
#any true value will do, if this line is reached the or do below won't run
1;
} or do {
#if we are in this block something bad happened, so we need to find out
#what it was by looking at the $@ variable which holds the reason the
#the code above died
if ($@ eq "timeout\n") {
#if $@ is the timeout message then we timed out, take the
#proper clean up steps here
} else {
#we died for some other reason, possibly a syntax error or the code
#issued its own die. We should look at $@ more carefully and determine
#the right course of action. For the purposes of this example I will
#assume that a message of "resource not available\n" is being sent by
#the thing that takes a long time and it is safe to continue the program.
#Any other message is unexpected.
#since we didn't timeout, but the code died, alarm 0 was never called
#so we need to call it now to prevent the default ALRM signal handler
#from running when the timeout is up
alarm 0;
if ($@ eq "resource not available\n") {
warn $@;
} else {
die $@;
}
}
或写得更紧凑:
eval {
local $SIG{ALRM} = sub { die "timeout\n" };
alarm $wait; #wait $wait seconds and then send the ALRM signal
#thing that could take a long time
alarm 0;
1;
} or do {
die $@ unless $@ eq "timeout\n" or $@ eq "resource not available\n";
alarm 0;
warn $@;
}