Perl:Windows 上的 exec() 与 Unix 上的比较

And*_*dyH 2 perl exec

下面的代码按照我在 Unix 上的预期工作 - system() 调用会阻塞,直到子进程完成。在 Windows 上,尽管它的工作方式不同 - system() 仅阻塞,直到子脚本到达 exec(),然后它立即返回,并且 exec-ed 子脚本继续在后台运行。有没有办法让它像在 Unix 上一样工作?

# main script
my $myCmd = [$^X, 'myScript.pl', 'arg1', 'arg2'];
system($^X, 'runcmd.pl', @$myCmd);


# runcmd.pl
open(STDOUT, '>', 'out.tmp');
open(STDERR, '>', 'err.tmp');
my $exe = shift @ARGV;
unless(exec($exe, @ARGV))
{
    close(STDOUT);
    close(STDERR);
    exit(1);
}

# myScript.pl - any script that runs few seconds and produces some output, e.g:
foreach (1..5)
{
    sleep 1;
    print "$_\n";
}
Run Code Online (Sandbox Code Playgroud)

ike*_*ami 5

不完全是,不。简单来说,exec就是改变当前进程正在执行的程序。这不是 Windows 支持的东西。

Perl 显然是exec通过在新进程中执行程序并退出现有进程来进行模拟的。就像在 UNIX 系统上一样,这使得单个程序运行。

exec在 Windows 上您可以通过替换为来获得所需的结果system

或者,完全避免第二个程序。IPC::RunIPC::Run3是用于启动其他程序的可靠模块。

use IPC::Run qw( run );

run [ $^X, 'myScript.pl', 'arg1', 'arg2' ],
   '>', "out.tmp",
   '2>', "err.tmp";
Run Code Online (Sandbox Code Playgroud)

即使是低级但核心的IPC::Open3也可以在这里工作。

use IPC::Run qw( open3 );

open(local *CHILD_STDOUT, '>', 'out.tmp') or die $!;
open(local *CHILD_STDERR, '>', 'err.tmp') or die $!;

my $pid = open3('<&STDIN', '>&CHILD_STDOUT', '>&CHILD_STDERR',
   $^X, 'myScript.pl', 'arg1', 'arg2');

waitpid($pid, 0);
Run Code Online (Sandbox Code Playgroud)