Question

I am trying change the stdout of the system function using select. But it does not seem to be working. The system output is getting displayed on console rather than getting redirected to the file.

use strict;
use warnings;

chdir "C:\\Documents and Settings\\" or die "cannot open the dir\n";
open FH, ">out.txt" or die "cannot open out.txt\n";
select FH or die " cannot change the stdout\n";
system "dir /a" ; 

Although I can do system "dir /a > out.txt", I want to know why the above code is not working.

Was it helpful?

Solution

The select function changes the default filehandle for output. It does not change STDOUT, which points to whatever it points to regardless if it is the default filehandle for output. If you want to change STDOUT, you have to do something like ylebre or Jon's answer.

When you start a child process, it gets the same standard output as the parent. It doesn't care what the default filehandle of the parent is though.

OTHER TIPS

The problem with your code is that system command doesn't capture the output for you. You can use qx//:

use strict;
use warnings;

open my $fh, ">", "output.txt" or die $!;
my $oldfh= select $fh;
print qx/ls -a/;
select $oldfh;
close $fh or warn $!;

There are other options to run an external application and obtain read it's output, like the IPC::Open2 and IPC::Open3 modules. They're included in perl by default.

A side note: don't use globs for file handles: they are globals, and globals are evil. Use lexical variables instead (FH vs. $fh). Also, you should use the three argument open instead of two parameters, since the former is safer than the latter.

Maybe this helps

I'm not sure 'select' is what you are looking for in this case. To redirect STDOUT to a file, you can do the following:

use strict;
use warnings;
use IO::Handle;
open FH, "> out.txt";
STDOUT->fdopen( \*FH, 'w' ) or die $!;
print "Hello world!\n";
system "dir /a";

Hope this helps!

I reassign the file descriptors at the system (not Perl) level:

    sub reopen($$$) {
        open $_[0], $_[1]. '&='. fileno($_[2]) or die "failed to reopen: $!";
    }

    reopen *STDOUT,  '>', $new_stdout_handle;

Now anything that you fork will think $new_stdout_handle is standard output.

I pasted a full example to gist: http://gist.github.com/129407

Licensed under: CC-BY-SA with attribution
Not affiliated with StackOverflow
scroll top