我正在编写我的第一个Perl应用程序 - 一个与Arduino微控制器对话的AOL Instant Messenger机器人,后者又控制一个伺服器,它将按下我们系统管理员服务器上的电源按钮,该服务器每28小时随机冻结一次。
我已经完成了所有艰难的工作,我只是尝试添加最后一段代码来打破主循环并在用户输入'quit'时退出AIM。
问题是,如果我尝试在主程序循环中读取STDIN,它会阻止进程直到输入输入,实质上是使机器人无效。我在阅读之前尝试过测试EOF,但没有骰子...... EOF总是会返回false。
以下是我正在使用的一些示例代码:
while(1) {
$oscar->do_one_loop();
# Poll to see if any arduino data is coming in over serial port
my $char = $port->lookfor();
# If we get data from arduino, then print it
if ($char) {
print "" . $char ;
}
# reading STDIN blocks until input is received... AAARG!
my $a = <STDIN>;
print $a;
if($a eq "exit" || $a eq "quit" || $a eq 'c' || $a eq 'q') {last;}
}
print "Signing off... ";
$oscar->signoff();
print "Done\n";
print "Closing serial port... ";
$port->close() || warn "close failed";
print "Done\n";
答案 0 :(得分:19)
内置的Perl是select()
,它是select()
系统调用的传递,但对于理智的人,我建议使用IO::Select
。
代码示例:
#!/usr/bin/perl
use IO::Select;
$s = IO::Select->new();
$s->add(\*STDIN);
while (++$i) {
print "Hiya $i!\n";
sleep(5);
if ($s->can_read(.5)) {
chomp($foo = <STDIN>);
print "Got '$foo' from STDIN\n";
}
}
答案 1 :(得分:1)
我发现,只要关闭STDOUT(例如,管道中的上游进程退出或输入来自文件),IO::Select就可以正常工作。但是,如果正在进行输出(例如从“ tail -f”输出),则不会显示<STDIN>
缓冲的任何部分数据。而是使用无缓冲的sysread:
#!/usr/bin/perl
use IO::Select;
$s = IO::Select->new(\*STDIN);
while (++$i) {
if ($s->can_read(2)) {
last unless defined($foo=get_unbuf_line());
print "Got '$foo'\n";
}
}
sub get_unbuf_line {
my $line="";
while (sysread(STDIN, my $nextbyte, 1)) {
return $line if $nextbyte eq "\n";
$line .= $nextbyte;
}
return(undef);
}