简单的open3示例不起作用

时间:2012-06-29 16:13:21

标签: perl ipc ipcopen3

我正在尝试创建一个调用子perl脚本并通过管道进行交互的Master perl脚本。

我已经为主人编写了这段代码:

#!/usr/bin/env perl

use strict;
use warnings;

use IPC::Open3;

my @children;

for my $i ( 0 .. 4 ) {
    print "Master: " . $i . ", I summon you\n";

    $children[$i] = {};

    $children[$i]->{'pid'} = open3( my $CH_IN, my $CH_OUT, my $CH_ERR, 'perl child.pl -i ' . $i );

    $children[$i]->{'_STDIN'}  = $CH_IN;
    $children[$i]->{'_STDOUT'} = $CH_OUT;
    $children[$i]->{'_STDERR'} = $CH_ERR;

    my $line = readline $children[$i]->{'_STDOUT'};
    print $line ;

}

print "Master: Go fetch me the sacred crown\n";

for my $i ( 0 .. 4 ) {
    $children[$i]->{'_STDIN'}->write("fetch the sacred crown\n");
    my $line = readline $children[$i]->{'_STDIN'};
    print $line ;
}

print "Master: Thanks. Now die!!!\n";

for my $i ( 0 .. 4 ) {
    $children[$i]->{'_STDIN'}->write("die !!\n");
    my $line = readline $children[$i]->{'_STDIN'};
    print $line ;
}

这是给孩子的:

#!/usr/bin/env perl

use Getopt::Long ;

my $cmdline_id ;

GetOptions ('i=s' => \$cmdline_id) ;

my $id = $cmdline_id ;

exit 1 if !defined $id ;

print "I am $id, and I am awaken\n" ;

while(<STDIN>) {
    print STDOUT $id . ': Master ask me to ' . $_ ;

    if ($_ =~ /exit/oi) {
        exit 0 ;
    }
}

但是当我启动主人时,他只是在阅读孩子的回答时挂了。

关于我做错了什么,为什么?

2 个答案:

答案 0 :(得分:4)

你是suffering from buffering

在子进程开头附近说$|=1以允许子进程打印而不等待输出缓冲区填满。

答案 1 :(得分:0)

作为旁注,为第三个arg传递undef(就像你一样)不能做你想要的。您需要使用符号gensym初始化变量。

use Symbol qw( gensym );

my %child;
$child{pid} = open3(
    $child{'_STDIN' } = gensym,
    $child{'_STDOUT'} = gensym,
    $child{'_STDERR'} = gensym,
    'perl', 'child.pl', '-i' => $i
);

$children[$i] = \%child;