如何在Perl中不阻塞地测试STDIN?

时间:2008-08-18 03:02:11

标签: perl stdin

我正在编写我的第一个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";

2 个答案:

答案 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);
}